2 * Copyright (C) 2007 Kevin Ollivier All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 #include "EventHandler.h"
32 #include "FocusController.h"
34 #include "FrameLoader.h"
35 #include "FrameView.h"
36 #include "GraphicsContext.h"
37 #include "HTMLFrameOwnerElement.h"
41 #include "PlatformKeyboardEvent.h"
42 #include "PlatformMouseEvent.h"
43 #include "PlatformString.h"
44 #include "PlatformWheelEvent.h"
45 #include "RenderObject.h"
46 #include "RenderTreeAsText.h"
47 #include "RenderView.h"
48 #include "SelectionController.h"
50 #include "SubstituteData.h"
52 #include "ChromeClientWx.h"
53 #include "ContextMenuClientWx.h"
54 #include "DragClientWx.h"
55 #include "EditorClientWx.h"
56 #include "FrameLoaderClientWx.h"
57 #include "InspectorClientWx.h"
59 #include "kjs_proxy.h"
60 #include "kjs_binding.h"
61 #include <kjs/value.h>
62 #include <kjs/ustring.h>
64 #include "wx/wxprec.h"
70 #include "WebViewPrivate.h"
73 #include <wx/dcbuffer.h>
75 // Match Safari's min/max zoom sizes by default
76 #define MinimumTextSizeMultiplier 0.5f
77 #define MaximumTextSizeMultiplier 3.0f
78 #define TextSizeMultiplierRatio 1.2f
84 return (int)(val < 0 ? val - 0.5 : val + 0.5);
88 // ----------------------------------------------------------------------------
90 // ----------------------------------------------------------------------------
92 IMPLEMENT_DYNAMIC_CLASS(wxWebViewLoadEvent, wxCommandEvent)
94 DEFINE_EVENT_TYPE(wxEVT_WEBVIEW_LOAD)
96 wxWebViewLoadEvent::wxWebViewLoadEvent(wxWindow* win)
98 SetEventType( wxEVT_WEBVIEW_LOAD);
99 SetEventObject( win );
104 IMPLEMENT_DYNAMIC_CLASS(wxWebViewBeforeLoadEvent, wxCommandEvent)
106 DEFINE_EVENT_TYPE(wxEVT_WEBVIEW_BEFORE_LOAD)
108 wxWebViewBeforeLoadEvent::wxWebViewBeforeLoadEvent(wxWindow* win)
111 SetEventType(wxEVT_WEBVIEW_BEFORE_LOAD);
117 IMPLEMENT_DYNAMIC_CLASS(wxWebViewNewWindowEvent, wxCommandEvent)
119 DEFINE_EVENT_TYPE(wxEVT_WEBVIEW_NEW_WINDOW)
121 wxWebViewNewWindowEvent::wxWebViewNewWindowEvent(wxWindow* win)
123 SetEventType(wxEVT_WEBVIEW_NEW_WINDOW);
129 IMPLEMENT_DYNAMIC_CLASS(wxWebViewRightClickEvent, wxCommandEvent)
131 DEFINE_EVENT_TYPE(wxEVT_WEBVIEW_RIGHT_CLICK)
133 wxWebViewRightClickEvent::wxWebViewRightClickEvent(wxWindow* win)
135 SetEventType(wxEVT_WEBVIEW_RIGHT_CLICK);
141 IMPLEMENT_DYNAMIC_CLASS(wxWebViewConsoleMessageEvent, wxCommandEvent)
143 DEFINE_EVENT_TYPE(wxEVT_WEBVIEW_CONSOLE_MESSAGE)
145 wxWebViewConsoleMessageEvent::wxWebViewConsoleMessageEvent(wxWindow* win)
147 SetEventType(wxEVT_WEBVIEW_CONSOLE_MESSAGE);
153 //---------------------------------------------------------
154 // DOM Element info data type
155 //---------------------------------------------------------
157 wxWebViewDOMElementInfo::wxWebViewDOMElementInfo() :
160 m_text(wxEmptyString),
161 m_imageSrc(wxEmptyString),
162 m_link(wxEmptyString)
166 BEGIN_EVENT_TABLE(wxWebView, wxWindow)
167 EVT_PAINT(wxWebView::OnPaint)
168 EVT_SIZE(wxWebView::OnSize)
169 EVT_MOUSE_EVENTS(wxWebView::OnMouseEvents)
170 EVT_KEY_DOWN(wxWebView::OnKeyEvents)
171 EVT_KEY_UP(wxWebView::OnKeyEvents)
172 EVT_CHAR(wxWebView::OnKeyEvents)
173 EVT_SET_FOCUS(wxWebView::OnSetFocus)
174 EVT_KILL_FOCUS(wxWebView::OnKillFocus)
175 EVT_ACTIVATE(wxWebView::OnActivate)
178 wxWebView::wxWebView(wxWindow* parent, int id, const wxPoint& position,
179 const wxSize& size, WebViewFrameData* data) :
180 m_textMagnifier(1.0),
182 m_isInitialized(false),
183 m_beingDestroyed(false),
184 m_title(wxEmptyString)
186 if (!wxWindow::Create(parent, id, position, size, wxBORDER_NONE | wxHSCROLL | wxVSCROLL))
189 // This is necessary because we are using SharedTimerWin.cpp on Windows,
190 // due to a problem with exceptions getting eaten when using the callback
191 // approach to timers (which wx itself uses).
193 WebCore::Page::setInstanceHandle(wxGetInstance());
196 // this helps reduce flicker on platforms like MSW
197 SetBackgroundStyle(wxBG_STYLE_CUSTOM);
199 m_impl = new WebViewPrivate();
201 WebCore::InitializeLoggingChannelsIfNecessary();
202 WebCore::HTMLFrameOwnerElement* parentFrame = 0;
204 // FIXME: This cast is obviously not as safe as a dynamic
205 // cast, but this allows us to get around requiring RTTI
206 // support for the moment. This is only used for subframes
207 // in any case, which aren't currently supported.
208 wxWebView* parentWebView = static_cast<wxWebView*>(parent);
211 parentFrame = data->ownerElement;
212 m_impl->page = parentWebView->m_impl->frame->page();
215 WebCore::EditorClientWx* editorClient = new WebCore::EditorClientWx();
216 m_impl->page = new WebCore::Page(new WebCore::ChromeClientWx(this), new WebCore::ContextMenuClientWx(), editorClient, new WebCore::DragClientWx(), new WebCore::InspectorClientWx());
217 editorClient->setPage(m_impl->page);
220 WebCore::FrameLoaderClientWx* loaderClient = new WebCore::FrameLoaderClientWx();
222 m_impl->frame = new WebCore::Frame(m_impl->page, parentFrame, loaderClient);
223 m_impl->frame->deref();
224 m_impl->frameView = new WebCore::FrameView(m_impl->frame.get());
225 m_impl->frameView->deref();
227 m_impl->frame->setView(m_impl->frameView.get());
228 m_impl->frame->init();
230 m_impl->frameView->setNativeWindow(this);
231 loaderClient->setFrame(m_impl->frame.get());
233 // Default settings - we should have wxWebViewSettings class for this
235 WebCore::Settings* settings = m_impl->page->settings();
236 settings->setLoadsImagesAutomatically(true);
237 settings->setDefaultFixedFontSize(13);
238 settings->setDefaultFontSize(16);
239 settings->setSerifFontFamily("Times New Roman");
240 settings->setFixedFontFamily("Courier New");
241 settings->setSansSerifFontFamily("Arial");
242 settings->setStandardFontFamily("Times New Roman");
243 settings->setJavaScriptEnabled(true);
245 m_isInitialized = true;
248 wxWebView::~wxWebView()
250 m_beingDestroyed = true;
252 m_impl->frame->loader()->detachFromParent();
256 // Since frameView has the last reference to Frame, once it is
257 // destroyed the destructor for Frame will happen as well.
258 m_impl->frameView = 0;
261 void wxWebView::Stop()
263 if (m_impl->frame && m_impl->frame->loader())
264 m_impl->frame->loader()->stop();
267 void wxWebView::Reload()
269 if (m_impl->frame && m_impl->frame->loader())
270 m_impl->frame->loader()->reload();
273 wxString wxWebView::GetPageSource()
276 if (m_impl->frameView && m_impl->frameView->layoutPending())
277 m_impl->frameView->layout();
279 WebCore::Document* doc = m_impl->frame->document();
282 wxString source = doc->toString();
286 return wxEmptyString;
289 void wxWebView::SetPageSource(const wxString& source, const wxString& baseUrl)
291 if (m_impl->frame && m_impl->frame->loader()) {
292 WebCore::FrameLoader* loader = m_impl->frame->loader();
293 loader->begin(WebCore::KURL(static_cast<const char*>(baseUrl.mb_str(wxConvUTF8))));
294 loader->write(source);
299 wxString wxWebView::GetInnerText()
301 if (m_impl->frameView && m_impl->frameView->layoutPending())
302 m_impl->frameView->layout();
304 WebCore::Element *documentElement = m_impl->frame->document()->documentElement();
305 return documentElement->innerText();
308 wxString wxWebView::GetAsMarkup()
310 if (!m_impl->frame || !m_impl->frame->document())
311 return wxEmptyString;
313 return createMarkup(m_impl->frame->document());
316 wxString wxWebView::GetExternalRepresentation()
318 if (m_impl->frameView && m_impl->frameView->layoutPending())
319 m_impl->frameView->layout();
321 return externalRepresentation(m_impl->frame->contentRenderer());
324 wxString wxWebView::RunScript(const wxString& javascript)
326 wxString returnValue = wxEmptyString;
328 KJS::JSValue* result = m_impl->frame->loader()->executeScript(javascript, true);
330 returnValue = wxString(result->toString(m_impl->frame->scriptProxy()->globalObject()->globalExec()).UTF8String().c_str(), wxConvUTF8);
335 void wxWebView::LoadURL(wxString url)
337 if (m_impl->frame && m_impl->frame->loader()) {
338 WebCore::KURL kurl = WebCore::KURL(static_cast<const char*>(url.mb_str(wxConvUTF8)));
339 // NB: This is an ugly fix, but CURL won't load sub-resources if the
340 // protocol is omitted; sadly, it will not emit an error, either, so
341 // there's no way for us to catch this problem the correct way yet.
342 if (kurl.protocol().isEmpty()) {
343 // is it a file on disk?
344 if (wxFileExists(url)) {
345 kurl.setProtocol("file");
346 kurl.setPath("//" + kurl.path());
349 kurl.setProtocol("http");
350 kurl.setPath("//" + kurl.path());
353 m_impl->frame->loader()->load(kurl);
357 bool wxWebView::GoBack()
359 if (m_impl->frame && m_impl->frame->page())
360 return m_impl->frame->page()->goBack();
365 bool wxWebView::GoForward()
367 if (m_impl->frame && m_impl->frame->page())
368 return m_impl->frame->page()->goForward();
373 bool wxWebView::CanGoBack()
375 if (m_impl->frame && m_impl->frame->page() && m_impl->frame->page()->backForwardList())
376 return m_impl->frame->page()->backForwardList()->backItem() != NULL;
381 bool wxWebView::CanGoForward()
383 if (m_impl->frame && m_impl->frame->page() && m_impl->frame->page()->backForwardList())
384 return m_impl->frame->page()->backForwardList()->forwardItem() != NULL;
389 bool wxWebView::CanIncreaseTextSize() const
392 if (m_textMagnifier*TextSizeMultiplierRatio <= MaximumTextSizeMultiplier)
398 void wxWebView::IncreaseTextSize()
400 if (CanIncreaseTextSize()) {
401 m_textMagnifier = m_textMagnifier*TextSizeMultiplierRatio;
402 m_impl->frame->setZoomFactor(m_textMagnifier, true);
406 bool wxWebView::CanDecreaseTextSize() const
409 if (m_textMagnifier/TextSizeMultiplierRatio >= MinimumTextSizeMultiplier)
415 void wxWebView::DecreaseTextSize()
417 if (CanDecreaseTextSize()) {
418 m_textMagnifier = m_textMagnifier/TextSizeMultiplierRatio;
419 m_impl->frame->setZoomFactor(m_textMagnifier, true);
423 void wxWebView::MakeEditable(bool enable)
425 m_isEditable = enable;
430 * Event forwarding functions to send events down to WebCore.
433 void wxWebView::OnPaint(wxPaintEvent& event)
435 if (m_beingDestroyed || !m_impl->frameView || !m_impl->frame)
438 wxAutoBufferedPaintDC dc(this);
440 if (IsShown() && m_impl->frame && m_impl->frame->document()) {
446 wxRect paintRect = GetUpdateRegion().GetBox();
448 WebCore::IntSize offset = m_impl->frameView->scrollOffset();
450 gcdc.SetDeviceOrigin(-offset.width(), -offset.height());
452 dc.SetDeviceOrigin(-offset.width(), -offset.height());
453 paintRect.Offset(offset.width(), offset.height());
456 WebCore::GraphicsContext* gc = new WebCore::GraphicsContext(&gcdc);
458 WebCore::GraphicsContext* gc = new WebCore::GraphicsContext((wxWindowDC*)&dc);
460 if (gc && m_impl->frame->contentRenderer()) {
461 if (m_impl->frameView->needsLayout())
462 m_impl->frameView->layout();
464 m_impl->frame->paint(gc, paintRect);
470 void wxWebView::OnSize(wxSizeEvent& event)
472 if (m_isInitialized && m_impl->frame && m_impl->frameView) {
473 m_impl->frame->sendResizeEvent();
474 m_impl->frameView->layout();
481 void wxWebView::OnMouseEvents(wxMouseEvent& event)
485 if (!m_impl->frame && m_impl->frameView)
488 wxPoint globalPoint = ClientToScreen(event.GetPosition());
490 wxEventType type = event.GetEventType();
492 if (type == wxEVT_MOUSEWHEEL) {
493 WebCore::PlatformWheelEvent wkEvent(event, globalPoint);
494 m_impl->frame->eventHandler()->handleWheelEvent(wkEvent);
498 WebCore::PlatformMouseEvent wkEvent(event, globalPoint);
500 if (type == wxEVT_LEFT_DOWN || type == wxEVT_MIDDLE_DOWN || type == wxEVT_RIGHT_DOWN)
501 m_impl->frame->eventHandler()->handleMousePressEvent(wkEvent);
503 else if (type == wxEVT_LEFT_UP || type == wxEVT_MIDDLE_UP || type == wxEVT_RIGHT_UP ||
504 type == wxEVT_LEFT_DCLICK || type == wxEVT_MIDDLE_DCLICK || type == wxEVT_RIGHT_DCLICK)
505 m_impl->frame->eventHandler()->handleMouseReleaseEvent(wkEvent);
507 else if (type == wxEVT_MOTION)
508 m_impl->frame->eventHandler()->handleMouseMoveEvent(wkEvent);
511 bool wxWebView::CanCopy()
513 if (m_impl->frame && m_impl->frameView)
514 return (m_impl->frame->editor()->canCopy() || m_impl->frame->editor()->canDHTMLCopy());
519 void wxWebView::Copy()
522 m_impl->frame->editor()->copy();
525 bool wxWebView::CanCut()
527 if (m_impl->frame && m_impl->frameView)
528 return (m_impl->frame->editor()->canCut() || m_impl->frame->editor()->canDHTMLCut());
533 void wxWebView::Cut()
536 m_impl->frame->editor()->cut();
539 bool wxWebView::CanPaste()
541 if (m_impl->frame && m_impl->frameView)
542 return (m_impl->frame->editor()->canPaste() || m_impl->frame->editor()->canDHTMLPaste());
547 void wxWebView::Paste()
550 m_impl->frame->editor()->paste();
554 void wxWebView::OnKeyEvents(wxKeyEvent& event)
556 if (m_impl->frame && m_impl->frameView) {
557 // WebCore doesn't handle these events itself, so we need to do
558 // it and not send the event down or else CTRL+C will erase the text
559 // and replace it with c.
560 if (event.CmdDown() && event.GetKeyCode() == static_cast<int>('C'))
562 else if (event.CmdDown() && event.GetKeyCode() == static_cast<int>('X'))
564 else if (event.CmdDown() && event.GetKeyCode() == static_cast<int>('V'))
567 WebCore::PlatformKeyboardEvent wkEvent(event);
568 if (wkEvent.type() == WebCore::PlatformKeyboardEvent::Char && wkEvent.altKey())
569 m_impl->frame->eventHandler()->handleAccessKey(wkEvent);
571 m_impl->frame->eventHandler()->keyEvent(wkEvent);
575 // make sure we get the character event.
576 if (event.GetEventType() != wxEVT_CHAR)
580 void wxWebView::OnSetFocus(wxFocusEvent& event)
583 m_impl->frame->selectionController()->setFocused(true);
588 void wxWebView::OnKillFocus(wxFocusEvent& event)
591 m_impl->frame->selectionController()->setFocused(false);
596 void wxWebView::OnActivate(wxActivateEvent& event)
599 m_impl->page->focusController()->setActive(event.GetActive());