]> git.saurik.com Git - wxWidgets.git/blob - src/msw/webview_ie.cpp
Add some version checks to help compiling on OSX.
[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, 2011 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 #include "wx/msw/registry.h"
27 #include "wx/msw/missing.h"
28 #include "wx/filesys.h"
29
30 wxIMPLEMENT_DYNAMIC_CLASS(wxWebViewIE, wxWebView);
31
32 //We link to urlmon as it is required for CoInternetGetSession
33 #pragma comment(lib, "urlmon")
34
35 BEGIN_EVENT_TABLE(wxWebViewIE, wxControl)
36 EVT_ACTIVEX(wxID_ANY, wxWebViewIE::onActiveXEvent)
37 EVT_ERASE_BACKGROUND(wxWebViewIE::onEraseBg)
38 END_EVENT_TABLE()
39
40 bool wxWebViewIE::Create(wxWindow* parent,
41 wxWindowID id,
42 const wxString& url,
43 const wxPoint& pos,
44 const wxSize& size,
45 long style,
46 const wxString& name)
47 {
48 if (!wxControl::Create(parent, id, pos, size, style,
49 wxDefaultValidator, name))
50 {
51 return false;
52 }
53
54 m_webBrowser = NULL;
55 m_isBusy = false;
56 m_historyLoadingFromList = false;
57 m_historyEnabled = true;
58 m_historyPosition = -1;
59 m_zoomType = wxWEB_VIEW_ZOOM_TYPE_TEXT;
60
61 if (::CoCreateInstance(CLSID_WebBrowser, NULL,
62 CLSCTX_INPROC_SERVER, // CLSCTX_INPROC,
63 IID_IWebBrowser2 , (void**)&m_webBrowser) != 0)
64 {
65 wxLogError("Failed to initialize IE, CoCreateInstance returned an error");
66 return false;
67 }
68
69 m_ie.SetDispatchPtr(m_webBrowser); // wxAutomationObject will release itself
70
71 m_webBrowser->put_RegisterAsBrowser(VARIANT_TRUE);
72 m_webBrowser->put_RegisterAsDropTarget(VARIANT_TRUE);
73
74 m_container = new wxActiveXContainer(this, IID_IWebBrowser2, m_webBrowser);
75
76 SetBackgroundStyle(wxBG_STYLE_PAINT);
77 SetDoubleBuffered(true);
78 LoadURL(url);
79 return true;
80 }
81
82
83 void wxWebViewIE::LoadURL(const wxString& url)
84 {
85 m_ie.CallMethod("Navigate", (BSTR) url.wc_str(), NULL, NULL, NULL, NULL);
86 }
87
88 void wxWebViewIE::SetPage(const wxString& html, const wxString& baseUrl)
89 {
90 BSTR bstr = SysAllocString(html.wc_str());
91
92 // Creates a new one-dimensional array
93 SAFEARRAY *psaStrings = SafeArrayCreateVector(VT_VARIANT, 0, 1);
94 if (psaStrings != NULL)
95 {
96 VARIANT *param;
97
98 HRESULT hr = SafeArrayAccessData(psaStrings, (LPVOID*)&param);
99 param->vt = VT_BSTR;
100 param->bstrVal = bstr;
101 hr = SafeArrayUnaccessData(psaStrings);
102
103 IHTMLDocument2* document = GetDocument();
104 document->write(psaStrings);
105 document->Release();
106
107 // SafeArrayDestroy calls SysFreeString for each BSTR
108 SafeArrayDestroy(psaStrings);
109
110 //We send the events when we are done to mimic webkit
111 //Navigated event
112 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED,
113 GetId(), baseUrl, "");
114 event.SetEventObject(this);
115 HandleWindowEvent(event);
116
117 //Document complete event
118 event.SetEventType(wxEVT_COMMAND_WEB_VIEW_LOADED);
119 event.SetEventObject(this);
120 HandleWindowEvent(event);
121 }
122 else
123 {
124 wxLogError("wxWebViewIE::SetPage() : psaStrings is NULL");
125 }
126
127 }
128
129 wxString wxWebViewIE::GetPageSource() const
130 {
131 IHTMLDocument2* document = GetDocument();
132 IHTMLElement *bodyTag = NULL;
133 IHTMLElement *htmlTag = NULL;
134 wxString source;
135 HRESULT hr = document->get_body(&bodyTag);
136 if(SUCCEEDED(hr))
137 {
138 hr = bodyTag->get_parentElement(&htmlTag);
139 if(SUCCEEDED(hr))
140 {
141 BSTR bstr;
142 htmlTag->get_outerHTML(&bstr);
143 source = wxString(bstr);
144 htmlTag->Release();
145 }
146 bodyTag->Release();
147 }
148
149 document->Release();
150 return source;
151 }
152
153 wxWebViewZoom wxWebViewIE::GetZoom() const
154 {
155 if(m_zoomType == wxWEB_VIEW_ZOOM_TYPE_LAYOUT)
156 return GetIEOpticalZoom();
157 else if(m_zoomType == wxWEB_VIEW_ZOOM_TYPE_TEXT)
158 return GetIETextZoom();
159 else
160 wxFAIL;
161
162 //Dummy return to stop compiler warnings
163 return wxWEB_VIEW_ZOOM_MEDIUM;
164
165 }
166
167 void wxWebViewIE::SetZoom(wxWebViewZoom zoom)
168 {
169 if(m_zoomType == wxWEB_VIEW_ZOOM_TYPE_LAYOUT)
170 SetIEOpticalZoom(zoom);
171 else if(m_zoomType == wxWEB_VIEW_ZOOM_TYPE_TEXT)
172 SetIETextZoom(zoom);
173 else
174 wxFAIL;
175 }
176
177 void wxWebViewIE::SetIETextZoom(wxWebViewZoom level)
178 {
179 //We do not use OLECMDID_OPTICAL_GETZOOMRANGE as the docs say the range
180 //is 0 to 4 so the check is unnecessary, these match exactly with the
181 //enum values
182 VARIANT zoomVariant;
183 VariantInit (&zoomVariant);
184 V_VT(&zoomVariant) = VT_I4;
185 V_I4(&zoomVariant) = level;
186
187 HRESULT result = m_webBrowser->ExecWB(OLECMDID_ZOOM,
188 OLECMDEXECOPT_DONTPROMPTUSER,
189 &zoomVariant, NULL);
190 wxASSERT(result == S_OK);
191 }
192
193 wxWebViewZoom wxWebViewIE::GetIETextZoom() const
194 {
195 VARIANT zoomVariant;
196 VariantInit (&zoomVariant);
197 V_VT(&zoomVariant) = VT_I4;
198
199 HRESULT result = m_webBrowser->ExecWB(OLECMDID_ZOOM,
200 OLECMDEXECOPT_DONTPROMPTUSER,
201 NULL, &zoomVariant);
202 wxASSERT(result == S_OK);
203
204 //We can safely cast here as we know that the range matches our enum
205 return static_cast<wxWebViewZoom>(V_I4(&zoomVariant));
206 }
207
208 void wxWebViewIE::SetIEOpticalZoom(wxWebViewZoom level)
209 {
210 //We do not use OLECMDID_OPTICAL_GETZOOMRANGE as the docs say the range
211 //is 10 to 1000 so the check is unnecessary
212 VARIANT zoomVariant;
213 VariantInit (&zoomVariant);
214 V_VT(&zoomVariant) = VT_I4;
215
216 //We make a somewhat arbitray map here, taken from values used by webkit
217 switch(level)
218 {
219 case wxWEB_VIEW_ZOOM_TINY:
220 V_I4(&zoomVariant) = 60;
221 break;
222 case wxWEB_VIEW_ZOOM_SMALL:
223 V_I4(&zoomVariant) = 80;
224 break;
225 case wxWEB_VIEW_ZOOM_MEDIUM:
226 V_I4(&zoomVariant) = 100;
227 break;
228 case wxWEB_VIEW_ZOOM_LARGE:
229 V_I4(&zoomVariant) = 130;
230 break;
231 case wxWEB_VIEW_ZOOM_LARGEST:
232 V_I4(&zoomVariant) = 160;
233 break;
234 default:
235 wxFAIL;
236 }
237
238 HRESULT result = m_webBrowser->ExecWB((OLECMDID)OLECMDID_OPTICAL_ZOOM,
239 OLECMDEXECOPT_DODEFAULT,
240 &zoomVariant,
241 NULL);
242 wxASSERT(result == S_OK);
243 }
244
245 wxWebViewZoom wxWebViewIE::GetIEOpticalZoom() const
246 {
247 VARIANT zoomVariant;
248 VariantInit (&zoomVariant);
249 V_VT(&zoomVariant) = VT_I4;
250
251 HRESULT result = m_webBrowser->ExecWB((OLECMDID)OLECMDID_OPTICAL_ZOOM,
252 OLECMDEXECOPT_DODEFAULT, NULL,
253 &zoomVariant);
254 wxASSERT(result == S_OK);
255
256 const int zoom = V_I4(&zoomVariant);
257
258 //We make a somewhat arbitray map here, taken from values used by webkit
259 if (zoom <= 65)
260 {
261 return wxWEB_VIEW_ZOOM_TINY;
262 }
263 else if (zoom > 65 && zoom <= 90)
264 {
265 return wxWEB_VIEW_ZOOM_SMALL;
266 }
267 else if (zoom > 90 && zoom <= 115)
268 {
269 return wxWEB_VIEW_ZOOM_MEDIUM;
270 }
271 else if (zoom > 115 && zoom <= 145)
272 {
273 return wxWEB_VIEW_ZOOM_LARGE;
274 }
275 else /*if (zoom > 145) */ //Using else removes a compiler warning
276 {
277 return wxWEB_VIEW_ZOOM_LARGEST;
278 }
279 }
280
281 void wxWebViewIE::SetZoomType(wxWebViewZoomType type)
282 {
283 m_zoomType = type;
284 }
285
286 wxWebViewZoomType wxWebViewIE::GetZoomType() const
287 {
288 return m_zoomType;
289 }
290
291 bool wxWebViewIE::CanSetZoomType(wxWebViewZoomType type) const
292 {
293 //IE 6 and below only support text zoom, so check the registry to see what
294 //version we actually have
295 wxRegKey key(wxRegKey::HKLM, "Software\\Microsoft\\Internet Explorer");
296 wxString value;
297 key.QueryValue("Version", value);
298
299 long version = wxAtoi(value.Left(1));
300 if(version <= 6 && type == wxWEB_VIEW_ZOOM_TYPE_LAYOUT)
301 return false;
302 else
303 return true;
304 }
305
306 void wxWebViewIE::Print()
307 {
308 m_webBrowser->ExecWB(OLECMDID_PRINTPREVIEW,
309 OLECMDEXECOPT_DODEFAULT, NULL, NULL);
310 }
311
312 bool wxWebViewIE::CanGoBack() const
313 {
314 if(m_historyEnabled)
315 return m_historyPosition > 0;
316 else
317 return false;
318 }
319
320 bool wxWebViewIE::CanGoForward() const
321 {
322 if(m_historyEnabled)
323 return m_historyPosition != static_cast<int>(m_historyList.size()) - 1;
324 else
325 return false;
326 }
327
328 void wxWebViewIE::LoadHistoryItem(wxSharedPtr<wxWebViewHistoryItem> item)
329 {
330 int pos = -1;
331 for(unsigned int i = 0; i < m_historyList.size(); i++)
332 {
333 //We compare the actual pointers to find the correct item
334 if(m_historyList[i].get() == item.get())
335 pos = i;
336 }
337 wxASSERT_MSG(pos != static_cast<int>(m_historyList.size()),
338 "invalid history item");
339 m_historyLoadingFromList = true;
340 LoadURL(item->GetUrl());
341 m_historyPosition = pos;
342 }
343
344 wxVector<wxSharedPtr<wxWebViewHistoryItem> > wxWebViewIE::GetBackwardHistory()
345 {
346 wxVector<wxSharedPtr<wxWebViewHistoryItem> > backhist;
347 //As we don't have std::copy or an iterator constructor in the wxwidgets
348 //native vector we construct it by hand
349 for(int i = 0; i < m_historyPosition; i++)
350 {
351 backhist.push_back(m_historyList[i]);
352 }
353 return backhist;
354 }
355
356 wxVector<wxSharedPtr<wxWebViewHistoryItem> > wxWebViewIE::GetForwardHistory()
357 {
358 wxVector<wxSharedPtr<wxWebViewHistoryItem> > forwardhist;
359 //As we don't have std::copy or an iterator constructor in the wxwidgets
360 //native vector we construct it by hand
361 for(int i = m_historyPosition + 1; i < static_cast<int>(m_historyList.size()); i++)
362 {
363 forwardhist.push_back(m_historyList[i]);
364 }
365 return forwardhist;
366 }
367
368 void wxWebViewIE::GoBack()
369 {
370 LoadHistoryItem(m_historyList[m_historyPosition - 1]);
371 }
372
373 void wxWebViewIE::GoForward()
374 {
375 LoadHistoryItem(m_historyList[m_historyPosition + 1]);
376 }
377
378 void wxWebViewIE::Stop()
379 {
380 m_ie.CallMethod("Stop");
381 }
382
383 void wxWebViewIE::ClearHistory()
384 {
385 m_historyList.clear();
386 m_historyPosition = -1;
387 }
388
389 void wxWebViewIE::EnableHistory(bool enable)
390 {
391 m_historyEnabled = enable;
392 m_historyList.clear();
393 m_historyPosition = -1;
394 }
395
396 void wxWebViewIE::Reload(wxWebViewReloadFlags flags)
397 {
398 VARIANTARG level;
399 VariantInit(&level);
400 V_VT(&level) = VT_I2;
401
402 switch(flags)
403 {
404 case wxWEB_VIEW_RELOAD_DEFAULT:
405 V_I2(&level) = REFRESH_NORMAL;
406 break;
407 case wxWEB_VIEW_RELOAD_NO_CACHE:
408 V_I2(&level) = REFRESH_COMPLETELY;
409 break;
410 default:
411 wxFAIL_MSG("Unexpected reload type");
412 }
413
414 m_webBrowser->Refresh2(&level);
415 }
416
417 bool wxWebViewIE::IsOfflineMode()
418 {
419 wxVariant out = m_ie.GetProperty("Offline");
420
421 wxASSERT(out.GetType() == "bool");
422
423 return out.GetBool();
424 }
425
426 void wxWebViewIE::SetOfflineMode(bool offline)
427 {
428 // FIXME: the wxWidgets docs do not really document what the return
429 // parameter of PutProperty is
430 const bool success = m_ie.PutProperty("Offline", (offline ?
431 VARIANT_TRUE :
432 VARIANT_FALSE));
433 wxASSERT(success);
434 }
435
436 bool wxWebViewIE::IsBusy() const
437 {
438 if (m_isBusy) return true;
439
440 wxVariant out = m_ie.GetProperty("Busy");
441
442 wxASSERT(out.GetType() == "bool");
443
444 return out.GetBool();
445 }
446
447 wxString wxWebViewIE::GetCurrentURL() const
448 {
449 wxVariant out = m_ie.GetProperty("LocationURL");
450
451 wxASSERT(out.GetType() == "string");
452 return out.GetString();
453 }
454
455 wxString wxWebViewIE::GetCurrentTitle() const
456 {
457 IHTMLDocument2* document = GetDocument();
458 BSTR title;
459
460 document->get_nameProp(&title);
461 document->Release();
462 return wxString(title);
463 }
464
465 bool wxWebViewIE::CanCut() const
466 {
467 return CanExecCommand("Cut");
468 }
469
470 bool wxWebViewIE::CanCopy() const
471 {
472 return CanExecCommand("Copy");
473 }
474 bool wxWebViewIE::CanPaste() const
475 {
476 return CanExecCommand("Paste");
477 }
478
479 void wxWebViewIE::Cut()
480 {
481 ExecCommand("Cut");
482 }
483
484 void wxWebViewIE::Copy()
485 {
486 ExecCommand("Copy");
487 }
488
489 void wxWebViewIE::Paste()
490 {
491 ExecCommand("Paste");
492 }
493
494 bool wxWebViewIE::CanUndo() const
495 {
496 return CanExecCommand("Undo");
497 }
498 bool wxWebViewIE::CanRedo() const
499 {
500 return CanExecCommand("Redo");
501 }
502
503 void wxWebViewIE::Undo()
504 {
505 ExecCommand("Undo");
506 }
507
508 void wxWebViewIE::Redo()
509 {
510 ExecCommand("Redo");
511 }
512
513 void wxWebViewIE::SetEditable(bool enable)
514 {
515 IHTMLDocument2* document = GetDocument();
516 if( enable )
517 document->put_designMode(SysAllocString(L"On"));
518 else
519 document->put_designMode(SysAllocString(L"Off"));
520
521 document->Release();
522 }
523
524 bool wxWebViewIE::IsEditable() const
525 {
526 IHTMLDocument2* document = GetDocument();
527 BSTR mode;
528 document->get_designMode(&mode);
529 document->Release();
530 if(wxString(mode) == "On")
531 return true;
532 else
533 return false;
534 }
535
536 void wxWebViewIE::SelectAll()
537 {
538 ExecCommand("SelectAll");
539 }
540
541 bool wxWebViewIE::HasSelection() const
542 {
543 IHTMLDocument2* document = GetDocument();
544 IHTMLSelectionObject* selection;
545 wxString sel;
546 HRESULT hr = document->get_selection(&selection);
547 if(SUCCEEDED(hr))
548 {
549 BSTR type;
550 selection->get_type(&type);
551 sel = wxString(type);
552 selection->Release();
553 }
554 document->Release();
555 return sel != "None";
556 }
557
558 void wxWebViewIE::DeleteSelection()
559 {
560 ExecCommand("Delete");
561 }
562
563 wxString wxWebViewIE::GetSelectedText() const
564 {
565 IHTMLDocument2* document = GetDocument();
566 IHTMLSelectionObject* selection;
567 wxString selected;
568 HRESULT hr = document->get_selection(&selection);
569 if(SUCCEEDED(hr))
570 {
571 IDispatch* disrange;
572 hr = selection->createRange(&disrange);
573 if(SUCCEEDED(hr))
574 {
575 IHTMLTxtRange* range;
576 hr = disrange->QueryInterface(IID_IHTMLTxtRange, (void**)&range);
577 if(SUCCEEDED(hr))
578 {
579 BSTR text;
580 range->get_text(&text);
581 selected = wxString(text);
582 range->Release();
583 }
584 disrange->Release();
585 }
586 selection->Release();
587 }
588 document->Release();
589 return selected;
590 }
591
592 wxString wxWebViewIE::GetSelectedSource() const
593 {
594 IHTMLDocument2* document = GetDocument();
595 IHTMLSelectionObject* selection;
596 wxString selected;
597 HRESULT hr = document->get_selection(&selection);
598 if(SUCCEEDED(hr))
599 {
600 IDispatch* disrange;
601 hr = selection->createRange(&disrange);
602 if(SUCCEEDED(hr))
603 {
604 IHTMLTxtRange* range;
605 hr = disrange->QueryInterface(IID_IHTMLTxtRange, (void**)&range);
606 if(SUCCEEDED(hr))
607 {
608 BSTR text;
609 range->get_htmlText(&text);
610 selected = wxString(text);
611 range->Release();
612 }
613 disrange->Release();
614 }
615 selection->Release();
616 }
617 document->Release();
618 return selected;
619 }
620
621 void wxWebViewIE::ClearSelection()
622 {
623 IHTMLDocument2* document = GetDocument();
624 IHTMLSelectionObject* selection;
625 wxString selected;
626 HRESULT hr = document->get_selection(&selection);
627 if(SUCCEEDED(hr))
628 {
629 selection->empty();
630 selection->Release();
631 }
632 document->Release();
633 }
634
635 wxString wxWebViewIE::GetPageText() const
636 {
637 IHTMLDocument2* document = GetDocument();
638 wxString text;
639 IHTMLElement* body;
640 HRESULT hr = document->get_body(&body);
641 if(SUCCEEDED(hr))
642 {
643 BSTR out;
644 body->get_innerText(&out);
645 text = wxString(out);
646 body->Release();
647 }
648 document->Release();
649 return text;
650 }
651
652 void wxWebViewIE::RunScript(const wxString& javascript)
653 {
654 IHTMLDocument2* document = GetDocument();
655 IHTMLWindow2* window;
656 wxString language = "javascript";
657 HRESULT hr = document->get_parentWindow(&window);
658 if(SUCCEEDED(hr))
659 {
660 VARIANT level;
661 VariantInit(&level);
662 V_VT(&level) = VT_EMPTY;
663 window->execScript(SysAllocString(javascript), SysAllocString(language), &level);
664 }
665 document->Release();
666 }
667
668 void wxWebViewIE::RegisterHandler(wxSharedPtr<wxWebViewHandler> handler)
669 {
670 ClassFactory* cf = new ClassFactory(handler);
671 IInternetSession* session;
672 if(FAILED(CoInternetGetSession(0, &session, 0)))
673 {
674 wxFAIL_MSG("Could not retrive internet session");
675 }
676
677 HRESULT hr = session->RegisterNameSpace(cf, CLSID_FileProtocol, handler->GetName(), 0, NULL, 0);
678 if(FAILED(hr))
679 {
680 wxFAIL_MSG("Could not register protocol");
681 }
682 }
683
684 bool wxWebViewIE::CanExecCommand(wxString command) const
685 {
686 IHTMLDocument2* document = GetDocument();
687 VARIANT_BOOL enabled;
688
689 document->queryCommandEnabled(SysAllocString(command.wc_str()), &enabled);
690 document->Release();
691
692 return (enabled == VARIANT_TRUE);
693 }
694
695 void wxWebViewIE::ExecCommand(wxString command)
696 {
697 IHTMLDocument2* document = GetDocument();
698 document->execCommand(SysAllocString(command.wc_str()), VARIANT_FALSE, VARIANT(), NULL);
699 document->Release();
700 }
701
702 IHTMLDocument2* wxWebViewIE::GetDocument() const
703 {
704 wxVariant variant = m_ie.GetProperty("Document");
705 IHTMLDocument2* document = (IHTMLDocument2*)variant.GetVoidPtr();
706
707 wxASSERT(document);
708
709 return document;
710 }
711
712 void wxWebViewIE::onActiveXEvent(wxActiveXEvent& evt)
713 {
714 if (m_webBrowser == NULL) return;
715
716 switch (evt.GetDispatchId())
717 {
718 case DISPID_BEFORENAVIGATE2:
719 {
720 m_isBusy = true;
721
722 wxString url = evt[1].GetString();
723 wxString target = evt[3].GetString();
724
725 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_NAVIGATING,
726 GetId(), url, target);
727 event.SetEventObject(this);
728 HandleWindowEvent(event);
729
730 if (!event.IsAllowed())
731 {
732 wxActiveXEventNativeMSW* nativeParams =
733 evt.GetNativeParameters();
734 *V_BOOLREF(&nativeParams->pDispParams->rgvarg[0]) = VARIANT_TRUE;
735 }
736
737 // at this point, either the navigation event has been cancelled
738 // and we're not busy, either it was accepted and IWebBrowser2's
739 // Busy property will be true; so we don't need our override
740 // flag anymore.
741 m_isBusy = false;
742
743 break;
744 }
745
746 case DISPID_NAVIGATECOMPLETE2:
747 {
748 wxString url = evt[1].GetString();
749 // TODO: set target parameter if possible
750 wxString target = wxEmptyString;
751 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_NAVIGATED,
752 GetId(), url, target);
753 event.SetEventObject(this);
754 HandleWindowEvent(event);
755 break;
756 }
757
758 case DISPID_PROGRESSCHANGE:
759 {
760 // download progress
761 break;
762 }
763
764 case DISPID_DOCUMENTCOMPLETE:
765 {
766 //Only send a complete even if we are actually finished, this brings
767 //the event in to line with webkit
768 READYSTATE rs;
769 m_webBrowser->get_ReadyState( &rs );
770 if(rs != READYSTATE_COMPLETE)
771 break;
772
773 wxString url = evt[1].GetString();
774
775 //As we are complete we also add to the history list, but not if the
776 //page is not the main page, ie it is a subframe
777 //We also have to check if we are loading a file:// url, if so we
778 //need to change the comparison as ie passes back a different style
779 //of url
780 if(m_historyEnabled && !m_historyLoadingFromList &&
781 (url == GetCurrentURL() ||
782 (GetCurrentURL().substr(0, 4) == "file" &&
783 wxFileSystem::URLToFileName(GetCurrentURL()).GetFullPath() == url)))
784 {
785 //If we are not at the end of the list, then erase everything
786 //between us and the end before adding the new page
787 if(m_historyPosition != static_cast<int>(m_historyList.size()) - 1)
788 {
789 m_historyList.erase(m_historyList.begin() + m_historyPosition + 1,
790 m_historyList.end());
791 }
792 wxSharedPtr<wxWebViewHistoryItem> item(new wxWebViewHistoryItem(url, GetCurrentTitle()));
793 m_historyList.push_back(item);
794 m_historyPosition++;
795 }
796 //Reset as we are done now
797 m_historyLoadingFromList = false;
798 // TODO: set target parameter if possible
799 wxString target = wxEmptyString;
800 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_LOADED, GetId(),
801 url, target);
802 event.SetEventObject(this);
803 HandleWindowEvent(event);
804 break;
805 }
806
807 case DISPID_STATUSTEXTCHANGE:
808 {
809 break;
810 }
811
812 case DISPID_TITLECHANGE:
813 {
814 wxString title = evt[0].GetString();
815
816 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_TITLE_CHANGED,
817 GetId(), GetCurrentURL(), "");
818 event.SetString(title);
819 event.SetEventObject(this);
820 HandleWindowEvent(event);
821 break;
822 }
823
824 case DISPID_NAVIGATEERROR:
825 {
826 wxWebViewNavigationError errorType = wxWEB_NAV_ERR_OTHER;
827 wxString errorCode = "?";
828 switch (evt[3].GetLong())
829 {
830 case INET_E_INVALID_URL: // (0x800C0002L or -2146697214)
831 errorCode = "INET_E_INVALID_URL";
832 errorType = wxWEB_NAV_ERR_REQUEST;
833 break;
834 case INET_E_NO_SESSION: // (0x800C0003L or -2146697213)
835 errorCode = "INET_E_NO_SESSION";
836 errorType = wxWEB_NAV_ERR_CONNECTION;
837 break;
838 case INET_E_CANNOT_CONNECT: // (0x800C0004L or -2146697212)
839 errorCode = "INET_E_CANNOT_CONNECT";
840 errorType = wxWEB_NAV_ERR_CONNECTION;
841 break;
842 case INET_E_RESOURCE_NOT_FOUND: // (0x800C0005L or -2146697211)
843 errorCode = "INET_E_RESOURCE_NOT_FOUND";
844 errorType = wxWEB_NAV_ERR_NOT_FOUND;
845 break;
846 case INET_E_OBJECT_NOT_FOUND: // (0x800C0006L or -2146697210)
847 errorCode = "INET_E_OBJECT_NOT_FOUND";
848 errorType = wxWEB_NAV_ERR_NOT_FOUND;
849 break;
850 case INET_E_DATA_NOT_AVAILABLE: // (0x800C0007L or -2146697209)
851 errorCode = "INET_E_DATA_NOT_AVAILABLE";
852 errorType = wxWEB_NAV_ERR_NOT_FOUND;
853 break;
854 case INET_E_DOWNLOAD_FAILURE: // (0x800C0008L or -2146697208)
855 errorCode = "INET_E_DOWNLOAD_FAILURE";
856 errorType = wxWEB_NAV_ERR_CONNECTION;
857 break;
858 case INET_E_AUTHENTICATION_REQUIRED: // (0x800C0009L or -2146697207)
859 errorCode = "INET_E_AUTHENTICATION_REQUIRED";
860 errorType = wxWEB_NAV_ERR_AUTH;
861 break;
862 case INET_E_NO_VALID_MEDIA: // (0x800C000AL or -2146697206)
863 errorCode = "INET_E_NO_VALID_MEDIA";
864 errorType = wxWEB_NAV_ERR_REQUEST;
865 break;
866 case INET_E_CONNECTION_TIMEOUT: // (0x800C000BL or -2146697205)
867 errorCode = "INET_E_CONNECTION_TIMEOUT";
868 errorType = wxWEB_NAV_ERR_CONNECTION;
869 break;
870 case INET_E_INVALID_REQUEST: // (0x800C000CL or -2146697204)
871 errorCode = "INET_E_INVALID_REQUEST";
872 errorType = wxWEB_NAV_ERR_REQUEST;
873 break;
874 case INET_E_UNKNOWN_PROTOCOL: // (0x800C000DL or -2146697203)
875 errorCode = "INET_E_UNKNOWN_PROTOCOL";
876 errorType = wxWEB_NAV_ERR_REQUEST;
877 break;
878 case INET_E_SECURITY_PROBLEM: // (0x800C000EL or -2146697202)
879 errorCode = "INET_E_SECURITY_PROBLEM";
880 errorType = wxWEB_NAV_ERR_SECURITY;
881 break;
882 case INET_E_CANNOT_LOAD_DATA: // (0x800C000FL or -2146697201)
883 errorCode = "INET_E_CANNOT_LOAD_DATA";
884 errorType = wxWEB_NAV_ERR_OTHER;
885 break;
886 case INET_E_CANNOT_INSTANTIATE_OBJECT:
887 // CoCreateInstance will return an error code if this happens,
888 // we'll handle this above.
889 return;
890 break;
891 case INET_E_REDIRECT_FAILED: // (0x800C0014L or -2146697196)
892 errorCode = "INET_E_REDIRECT_FAILED";
893 errorType = wxWEB_NAV_ERR_OTHER;
894 break;
895 case INET_E_REDIRECT_TO_DIR: // (0x800C0015L or -2146697195)
896 errorCode = "INET_E_REDIRECT_TO_DIR";
897 errorType = wxWEB_NAV_ERR_REQUEST;
898 break;
899 case INET_E_CANNOT_LOCK_REQUEST: // (0x800C0016L or -2146697194)
900 errorCode = "INET_E_CANNOT_LOCK_REQUEST";
901 errorType = wxWEB_NAV_ERR_OTHER;
902 break;
903 case INET_E_USE_EXTEND_BINDING: // (0x800C0017L or -2146697193)
904 errorCode = "INET_E_USE_EXTEND_BINDING";
905 errorType = wxWEB_NAV_ERR_OTHER;
906 break;
907 case INET_E_TERMINATED_BIND: // (0x800C0018L or -2146697192)
908 errorCode = "INET_E_TERMINATED_BIND";
909 errorType = wxWEB_NAV_ERR_OTHER;
910 break;
911 case INET_E_INVALID_CERTIFICATE: // (0x800C0019L or -2146697191)
912 errorCode = "INET_E_INVALID_CERTIFICATE";
913 errorType = wxWEB_NAV_ERR_CERTIFICATE;
914 break;
915 case INET_E_CODE_DOWNLOAD_DECLINED: // (0x800C0100L or -2146696960)
916 errorCode = "INET_E_CODE_DOWNLOAD_DECLINED";
917 errorType = wxWEB_NAV_ERR_USER_CANCELLED;
918 break;
919 case INET_E_RESULT_DISPATCHED: // (0x800C0200L or -2146696704)
920 // cancel request cancelled...
921 errorCode = "INET_E_RESULT_DISPATCHED";
922 errorType = wxWEB_NAV_ERR_OTHER;
923 break;
924 case INET_E_CANNOT_REPLACE_SFP_FILE: // (0x800C0300L or -2146696448)
925 errorCode = "INET_E_CANNOT_REPLACE_SFP_FILE";
926 errorType = wxWEB_NAV_ERR_SECURITY;
927 break;
928 case INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY:
929 errorCode = "INET_E_CODE_INSTALL_BLOCKED_BY_HASH_POLICY";
930 errorType = wxWEB_NAV_ERR_SECURITY;
931 break;
932 case INET_E_CODE_INSTALL_SUPPRESSED:
933 errorCode = "INET_E_CODE_INSTALL_SUPPRESSED";
934 errorType = wxWEB_NAV_ERR_SECURITY;
935 break;
936 }
937
938 wxString url = evt[1].GetString();
939 wxString target = evt[2].GetString();
940 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_ERROR, GetId(),
941 url, target);
942 event.SetEventObject(this);
943 event.SetInt(errorType);
944 event.SetString(errorCode);
945 HandleWindowEvent(event);
946 break;
947 }
948 case DISPID_NEWWINDOW3:
949 {
950 wxString url = evt[4].GetString();
951
952 wxWebViewEvent event(wxEVT_COMMAND_WEB_VIEW_NEWWINDOW,
953 GetId(), url, wxEmptyString);
954 event.SetEventObject(this);
955 HandleWindowEvent(event);
956
957 //We always cancel this event otherwise an Internet Exporer window
958 //is opened for the url
959 wxActiveXEventNativeMSW* nativeParams = evt.GetNativeParameters();
960 *V_BOOLREF(&nativeParams->pDispParams->rgvarg[3]) = VARIANT_TRUE;
961 break;
962 }
963 }
964
965 evt.Skip();
966 }
967
968 VirtualProtocol::VirtualProtocol(wxSharedPtr<wxWebViewHandler> handler)
969 {
970 m_refCount = 0;
971 m_file = NULL;
972 m_handler = handler;
973 }
974
975 VirtualProtocol::~VirtualProtocol()
976 {
977 }
978
979 ULONG VirtualProtocol::AddRef()
980 {
981 m_refCount++;
982 return m_refCount;
983 }
984
985 HRESULT VirtualProtocol::QueryInterface(REFIID riid, void **ppvObject)
986 {
987 if(riid == IID_IUnknown || riid == IID_IInternetProtocolRoot ||
988 riid == IID_IInternetProtocol)
989 {
990 *ppvObject = (IInternetProtocol*)this;
991 AddRef();
992 return S_OK;
993 }
994 else
995 {
996 *ppvObject = NULL;
997 return E_POINTER;
998 }
999 }
1000
1001 ULONG VirtualProtocol::Release()
1002 {
1003 m_refCount--;
1004 if (m_refCount > 0)
1005 {
1006 return m_refCount;
1007 }
1008 else
1009 {
1010 delete this;
1011 return 0;
1012 }
1013 }
1014
1015 HRESULT VirtualProtocol::Start(LPCWSTR szUrl, IInternetProtocolSink *pOIProtSink,
1016 IInternetBindInfo *pOIBindInfo, DWORD grfPI,
1017 HANDLE_PTR dwReserved)
1018 {
1019 wxUnusedVar(szUrl);
1020 wxUnusedVar(pOIBindInfo);
1021 wxUnusedVar(grfPI);
1022 wxUnusedVar(dwReserved);
1023 m_protocolSink = pOIProtSink;
1024
1025 //We get the file itself from the protocol handler
1026 m_file = m_handler->GetFile(szUrl);
1027
1028
1029 if(!m_file)
1030 return INET_E_RESOURCE_NOT_FOUND;
1031
1032 //We return the stream length for current and total size as we can always
1033 //read the whole file from the stream
1034 wxFileOffset length = m_file->GetStream()->GetLength();
1035 m_protocolSink->ReportData(BSCF_FIRSTDATANOTIFICATION |
1036 BSCF_DATAFULLYAVAILABLE |
1037 BSCF_LASTDATANOTIFICATION,
1038 length, length);
1039 return S_OK;
1040 }
1041
1042 HRESULT VirtualProtocol::Read(void *pv, ULONG cb, ULONG *pcbRead)
1043 {
1044 //If the file is null we return false to indicte it is finished
1045 if(!m_file)
1046 return S_FALSE;
1047
1048 wxStreamError err = m_file->GetStream()->Read(pv, cb).GetLastError();
1049 *pcbRead = m_file->GetStream()->LastRead();
1050
1051 if(err == wxSTREAM_NO_ERROR)
1052 {
1053 if(*pcbRead < cb)
1054 {
1055 wxDELETE(m_file);
1056 m_protocolSink->ReportResult(S_OK, 0, NULL);
1057 }
1058 //As we are not eof there is more data
1059 return S_OK;
1060 }
1061 else if(err == wxSTREAM_EOF)
1062 {
1063 wxDELETE(m_file);
1064 m_protocolSink->ReportResult(S_OK, 0, NULL);
1065 //We are eof and so finished
1066 return S_OK;
1067 }
1068 else if(err == wxSTREAM_READ_ERROR)
1069 {
1070 wxDELETE(m_file);
1071 return INET_E_DOWNLOAD_FAILURE;
1072 }
1073 else
1074 {
1075 //Dummy return to surpress a compiler warning
1076 wxFAIL;
1077 return INET_E_DOWNLOAD_FAILURE;
1078 }
1079 }
1080
1081 HRESULT ClassFactory::CreateInstance(IUnknown* pUnkOuter, REFIID riid,
1082 void ** ppvObject)
1083 {
1084 if (pUnkOuter)
1085 return CLASS_E_NOAGGREGATION;
1086 VirtualProtocol* vp = new VirtualProtocol(m_handler);
1087 vp->AddRef();
1088 HRESULT hr = vp->QueryInterface(riid, ppvObject);
1089 vp->Release();
1090 return hr;
1091
1092 }
1093
1094 STDMETHODIMP ClassFactory::LockServer(BOOL fLock)
1095 {
1096 wxUnusedVar(fLock);
1097 return S_OK;
1098 }
1099
1100 ULONG ClassFactory::AddRef(void)
1101 {
1102 m_refCount++;
1103 return m_refCount;
1104 }
1105
1106 HRESULT ClassFactory::QueryInterface(REFIID riid, void **ppvObject)
1107 {
1108 if ((riid == IID_IUnknown) || (riid == IID_IClassFactory))
1109 {
1110 *ppvObject = this;
1111 AddRef();
1112 return S_OK;
1113 }
1114 else
1115 {
1116 *ppvObject = NULL;
1117 return E_POINTER;
1118 }
1119
1120 }
1121
1122 ULONG ClassFactory::Release(void)
1123 {
1124 m_refCount--;
1125 if (m_refCount > 0)
1126 {
1127 return m_refCount;
1128 }
1129 else
1130 {
1131 delete this;
1132 return 0;
1133 }
1134
1135 }
1136
1137 #endif