/* Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies) This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #include "PluginContainerQt.h" #include "FocusController.h" #include "Frame.h" #include "FrameView.h" #include "Page.h" #include "PlatformKeyboardEvent.h" #include "PlatformWheelEvent.h" #include "PluginView.h" #include <QApplication> #include <QX11Info> using namespace WebCore; PluginClientWrapper::PluginClientWrapper(QWidget* parent, WId client) : QWidget(0, Qt::Popup) { // create a QWidget that adopts the plugin window id, do not give it // a parent so that we don't end up handling events supposed to be // handled by the QX11EmbedContainer. // without the parent this will be considered a toplevel widget, // and thus make Qt not quit the event loop after the last window // has been closed. In order to work around this, we set the window // type to Qt::Popup. create(client, false, true); m_parent = parent; } PluginClientWrapper::~PluginClientWrapper() { destroy(false, false); } bool PluginClientWrapper::x11Event(XEvent* event) { // modify the event window id and insert it into the Qt event system. event->xany.window = m_parent->effectiveWinId(); static_cast<QApplication*>(QApplication::instance())->x11ProcessEvent(event); return true; } PluginContainerQt::PluginContainerQt(PluginView* view, QWidget* parent) : QX11EmbedContainer(parent) , m_pluginView(view) , m_clientWrapper(0) { connect(this, SIGNAL(clientClosed()), this, SLOT(on_clientClosed())); connect(this, SIGNAL(clientIsEmbedded()), this, SLOT(on_clientIsEmbedded())); } PluginContainerQt::~PluginContainerQt() { delete m_clientWrapper; m_pluginView->setPlatformPluginWidget(0); } void PluginContainerQt::on_clientClosed() { delete m_clientWrapper; m_clientWrapper = 0; } void PluginContainerQt::on_clientIsEmbedded() { delete m_clientWrapper; m_clientWrapper = 0; // Only create a QWidget wrapper for the plugin in the case it isn't in the // Qt window mapper, and thus receiving events from the Qt event system. // This way the PluginClientWrapper receives the scroll events and passes // them to the parent. NOTICE: Native Qt based plugins running in process, // will already be in the window mapper, and thus creating a wrapper, stops // them from getting events from Qt, as they are redirected to the wrapper. if (!QWidget::find(clientWinId())) m_clientWrapper = new PluginClientWrapper(this, clientWinId()); } void PluginContainerQt::redirectWheelEventsToParent(bool enable) { // steal wheel events from the plugin as we want to handle it. When doing this // all button 4, 5, 6, and 7, ButtonPress and ButtonRelease events are passed // to the x11Event handler of the PluginClientWrapper, which then changes the // window id of the event to the parent of PluginContainer and puts the event // back into the Qt event loop, so that we will actually scroll the parent // frame. for (int buttonNo = 4; buttonNo < 8; buttonNo++) { if (enable) XGrabButton(x11Info().display(), buttonNo, AnyModifier, clientWinId(), false, ButtonPressMask, GrabModeAsync, GrabModeAsync, 0L, 0L); else XUngrabButton(x11Info().display(), buttonNo, AnyModifier, clientWinId()); } } bool PluginContainerQt::x11Event(XEvent* event) { switch (event->type) { case EnterNotify: // if the plugin window doesn't have focus we do not want to send wheel // events to it, but to the parent frame, so let's redirect here. redirectWheelEventsToParent(!hasFocus()); break; case LeaveNotify: // it is always safe to ungrab wheel events when the mouse leaves the // plugin window. redirectWheelEventsToParent(false); break; } return QX11EmbedContainer::x11Event(event); } void PluginContainerQt::focusInEvent(QFocusEvent* event) { // we got focus, stop redirecting the wheel events redirectWheelEventsToParent(false); if (Page* page = m_pluginView->parentFrame()->page()) page->focusController()->setActive(true); m_pluginView->focusPluginElement(); } void PluginContainerQt::focusOutEvent(QFocusEvent*) { if (Page* page = m_pluginView->parentFrame()->page()) page->focusController()->setActive(false); }