mirror of
https://bitbucket.org/smil3y/kdelibs.git
synced 2025-02-24 02:42:48 +00:00
1289 lines
46 KiB
C++
1289 lines
46 KiB
C++
/**
|
|
* This file is part of the HTML widget for KDE.
|
|
*
|
|
* Copyright (C) 1999-2003 Lars Knoll (knoll@kde.org)
|
|
* (C) 2000-2003 Dirk Mueller (mueller@kde.org)
|
|
* (C) 2003 Apple Computer, Inc.
|
|
* (C) 2004-2006 Germain Garand (germain@ebooksfrance.org)
|
|
*
|
|
* 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 "render_replaced.h"
|
|
#include "render_layer.h"
|
|
#include "render_canvas.h"
|
|
#include "render_line.h"
|
|
#include "rendering/render_position.h"
|
|
|
|
#include "render_arena.h"
|
|
|
|
#include <assert.h>
|
|
#include <QtGui/QWidget>
|
|
#include <QtGui/QPainter>
|
|
#include <QtGui/QActionEvent>
|
|
#include <QtGui/QApplication>
|
|
#include <QtGui/QLineEdit>
|
|
#include <QtGui/QComboBox>
|
|
#include <QtGui/QCheckBox>
|
|
#include <QtGui/QRadioButton>
|
|
#include <kglobalsettings.h>
|
|
#include <kurlrequester.h>
|
|
#include <QtCore/QObject>
|
|
#include <QVector>
|
|
#include <QMatrix>
|
|
#include <QtGui/QPaintEngine>
|
|
|
|
#include "khtml_ext.h"
|
|
#include "khtmlview.h"
|
|
#include "xml/dom2_eventsimpl.h"
|
|
#include "khtml_part.h"
|
|
#include "xml/dom_docimpl.h"
|
|
#include "xml/dom_position.h"
|
|
#include "misc/helper.h"
|
|
#include "misc/paintbuffer.h"
|
|
#include "css/cssvalues.h"
|
|
#include <kdebug.h>
|
|
|
|
//#define IN_PLACE_WIDGET_PAINTING
|
|
|
|
bool khtml::allowWidgetPaintEvents = false;
|
|
|
|
using namespace khtml;
|
|
using namespace DOM;
|
|
|
|
|
|
RenderReplaced::RenderReplaced(DOM::NodeImpl* node)
|
|
: RenderBox(node)
|
|
{
|
|
// init RenderObject attributes
|
|
setReplaced(true);
|
|
|
|
m_intrinsicWidth = 300;
|
|
m_intrinsicHeight = 150;
|
|
}
|
|
|
|
void RenderReplaced::calcMinMaxWidth()
|
|
{
|
|
KHTMLAssert( !minMaxKnown());
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
kDebug( 6040 ) << "RenderReplaced::calcMinMaxWidth() known=" << minMaxKnown();
|
|
#endif
|
|
|
|
m_width = calcReplacedWidth() + borderLeft() + borderRight() + paddingLeft() + paddingRight();
|
|
|
|
if ( style()->width().isPercent() || style()->height().isPercent() ||
|
|
style()->maxWidth().isPercent() || style()->maxHeight().isPercent() ||
|
|
style()->minWidth().isPercent() || style()->minHeight().isPercent() ) {
|
|
m_minWidth = 0;
|
|
m_maxWidth = m_width;
|
|
}
|
|
else
|
|
m_minWidth = m_maxWidth = m_width;
|
|
|
|
setMinMaxKnown();
|
|
}
|
|
|
|
FindSelectionResult RenderReplaced::checkSelectionPoint(int _x, int _y, int _tx, int _ty, DOM::NodeImpl*& node, int &offset, SelPointState &)
|
|
{
|
|
#if 0
|
|
kDebug(6040) << "RenderReplaced::checkSelectionPoint(_x="<<_x<<",_y="<<_y<<",_tx="<<_tx<<",_ty="<<_ty<<")" << endl
|
|
<< "xPos: " << xPos() << " yPos: " << yPos() << " width: " << width() << " height: " << height() << endl
|
|
<< "_ty + yPos: " << (_ty + yPos()) << " + height: " << (_ty + yPos() + height()) << "; _tx + xPos: " << (_tx + xPos()) << " + width: " << (_tx + xPos() + width()) << endl;
|
|
#endif
|
|
node = element();
|
|
offset = 0;
|
|
|
|
if ( _y < _ty + yPos() )
|
|
return SelectionPointBefore; // above -> before
|
|
|
|
if ( _y > _ty + yPos() + height() ) {
|
|
// below -> after
|
|
// Set the offset to the max
|
|
offset = 1;
|
|
return SelectionPointAfter;
|
|
}
|
|
if ( _x > _tx + xPos() + width() ) {
|
|
// to the right
|
|
// ### how to regard bidi in replaced elements? (LS)
|
|
offset = 1;
|
|
return SelectionPointAfterInLine;
|
|
}
|
|
|
|
// The Y matches, check if we're on the left
|
|
if ( _x < _tx + xPos() ) {
|
|
// ### how to regard bidi in replaced elements? (LS)
|
|
return SelectionPointBeforeInLine;
|
|
}
|
|
|
|
offset = _x > _tx + xPos() + width()/2;
|
|
return SelectionPointInside;
|
|
}
|
|
|
|
long RenderReplaced::caretMinOffset() const
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// Returns 1 since a replaced element can have the caret positioned
|
|
// at its beginning (0), or at its end (1).
|
|
long RenderReplaced::caretMaxOffset() const
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
unsigned long RenderReplaced::caretMaxRenderedOffset() const
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
RenderPosition RenderReplaced::positionForCoordinates(int _x, int _y)
|
|
{
|
|
InlineBox *box = placeHolderBox();
|
|
if (!box)
|
|
return RenderPosition(element(), 0);
|
|
|
|
RootInlineBox *root = box->root();
|
|
|
|
int absx, absy;
|
|
containingBlock()->absolutePosition(absx, absy);
|
|
|
|
int top = absy + root->topOverflow();
|
|
int bottom = root->nextRootBox() ? absy + root->nextRootBox()->topOverflow() : absy + root->bottomOverflow();
|
|
|
|
if (_y < top)
|
|
return RenderPosition(element(), caretMinOffset()); // coordinates are above
|
|
|
|
if (_y >= bottom)
|
|
return RenderPosition(element(), caretMaxOffset()); // coordinates are below
|
|
|
|
if (element()) {
|
|
if (_x <= absx + xPos() + (width() / 2))
|
|
return RenderPosition(element(), 0);
|
|
return RenderPosition(element(), 1);
|
|
}
|
|
|
|
return RenderBox::positionForCoordinates(_x, _y);
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
RenderWidget::RenderWidget(DOM::NodeImpl* node)
|
|
: RenderReplaced(node)
|
|
{
|
|
m_widget = 0;
|
|
m_underMouse = 0;
|
|
m_buffer[0] = 0;
|
|
m_buffer[1] = 0;
|
|
m_nativeFrameShape = QFrame::NoFrame;
|
|
// a widget doesn't support being anonymous
|
|
assert(!isAnonymous());
|
|
m_view = node->document()->view();
|
|
m_arena.reset(renderArena());
|
|
m_needsMask = false;
|
|
m_ownsWidget = true;
|
|
|
|
// this is no real reference counting, it is just there
|
|
// to make sure that we're not deleted while we're recursed
|
|
// in an eventFilter of the widget
|
|
ref();
|
|
}
|
|
|
|
void RenderWidget::detach()
|
|
{
|
|
// warning: keep in sync with RenderObject::detach
|
|
|
|
detachCounters();
|
|
remove();
|
|
|
|
if ( m_widget ) {
|
|
if ( m_view ) {
|
|
m_view->setWidgetVisible(this, false);
|
|
}
|
|
KHTMLWidget* k = dynamic_cast<KHTMLWidget*>(m_widget);
|
|
if (k)
|
|
k->m_kwp->setRenderWidget(0);
|
|
m_widget->removeEventFilter( this );
|
|
m_widget->setMouseTracking( false );
|
|
}
|
|
|
|
// make sure our DOM-node don't think we exist
|
|
if ( node() && node()->renderer() == this)
|
|
node()->setRenderer(0);
|
|
|
|
setDetached();
|
|
deref();
|
|
}
|
|
|
|
RenderWidget::~RenderWidget()
|
|
{
|
|
KHTMLAssert( refCount() <= 0 );
|
|
|
|
if(m_widget) {
|
|
if (m_widget->hasFocus ())
|
|
m_widget->clearFocus ();
|
|
m_widget->hide();
|
|
if (m_ownsWidget)
|
|
m_widget->deleteLater();
|
|
}
|
|
delete m_buffer[0];
|
|
delete m_buffer[1];
|
|
}
|
|
|
|
class QWidgetResizeEvent : public QEvent
|
|
{
|
|
public:
|
|
enum { Type = QEvent::User + 0xbee };
|
|
QWidgetResizeEvent( int _w, int _h ) :
|
|
QEvent( ( QEvent::Type ) Type ), w( _w ), h( _h ) {}
|
|
int w;
|
|
int h;
|
|
};
|
|
|
|
void RenderWidget::resizeWidget( int w, int h )
|
|
{
|
|
// ugly hack to limit the maximum size of the widget ( as X11 has problems if
|
|
// it is bigger )
|
|
h = qMin( h, 3072 );
|
|
w = qMin( w, 2000 );
|
|
|
|
if (m_widget->width() != w || m_widget->height() != h) {
|
|
m_widget->resize( w, h);
|
|
if (isRedirectedWidget() && qobject_cast<KHTMLView*>(m_widget) && !m_widget->isVisible()) {
|
|
// Emission of Resize event is delayed.
|
|
// we have to pre-call KHTMLView::resizeEvent
|
|
// so that viewport size change and subsequent layout update
|
|
// is effective synchronously, which is important for JS.
|
|
// This only work because m_widget is a redirected view,
|
|
// and thus has visibleWidth()/visibleHeight() that mirror this RenderWidget,
|
|
// rather than the effective widget size. - gg.
|
|
QResizeEvent e( QSize(w,h), QSize(m_widget->width(),m_widget->height()));
|
|
static_cast<KHTMLView*>(m_widget)->resizeEvent( &e );
|
|
}
|
|
}
|
|
}
|
|
|
|
bool RenderWidget::event( QEvent *e )
|
|
{
|
|
// eat all events - except if this is a frame (in which case KHTMLView handles it all)
|
|
if ( qobject_cast<KHTMLView*>( m_widget ) )
|
|
return QObject::event( e );
|
|
return true;
|
|
}
|
|
|
|
bool RenderWidget::isRedirectedWidget() const
|
|
{
|
|
KHTMLWidget* k = dynamic_cast<KHTMLWidget*>(m_widget);
|
|
return k ? k->m_kwp->isRedirected() : false;
|
|
}
|
|
|
|
void RenderWidget::setQWidget(QWidget *widget)
|
|
{
|
|
if (widget != m_widget)
|
|
{
|
|
if (m_widget) {
|
|
m_widget->removeEventFilter(this);
|
|
disconnect( m_widget, SIGNAL(destroyed()), this, SLOT(slotWidgetDestructed()));
|
|
m_widget->hide();
|
|
if (m_ownsWidget)
|
|
m_widget->deleteLater(); //Might happen due to event on the widget, so be careful
|
|
m_widget = 0;
|
|
}
|
|
m_widget = widget;
|
|
if (m_widget) {
|
|
KHTMLWidget* k = dynamic_cast<KHTMLWidget*>(m_widget);
|
|
bool isRedirectedSubFrame = false;
|
|
if (k) {
|
|
k->m_kwp->setRenderWidget(this);
|
|
// enable redirection of every sub-frame that is not a FRAME
|
|
if (qobject_cast<KHTMLView*>(m_widget) && element() && element()->id() != ID_FRAME) {
|
|
k->m_kwp->setIsRedirected( true );
|
|
isRedirectedSubFrame = true;
|
|
}
|
|
}
|
|
m_widget->setParent(m_view->widget());
|
|
if (isRedirectedSubFrame)
|
|
static_cast<KHTMLView*>(m_widget)->setHasStaticBackground();
|
|
connect( m_widget, SIGNAL(destroyed()), this, SLOT(slotWidgetDestructed()));
|
|
m_widget->installEventFilter(this);
|
|
if (isRedirectedWidget()) {
|
|
if (!qobject_cast<QFrame*>(m_widget))
|
|
m_widget->setAttribute( Qt::WA_NoSystemBackground );
|
|
}
|
|
if (m_widget->focusPolicy() > Qt::StrongFocus)
|
|
m_widget->setFocusPolicy(Qt::StrongFocus);
|
|
// if we've already received a layout, apply the calculated space to the
|
|
// widget immediately, but we have to have really been full constructed (with a non-null
|
|
// style pointer).
|
|
if (!needsLayout() && style()) {
|
|
resizeWidget( m_width-borderLeft()-borderRight()-paddingLeft()-paddingRight(),
|
|
m_height-borderTop()-borderBottom()-paddingTop()-paddingBottom() );
|
|
}
|
|
else
|
|
setPos(xPos(), -500000);
|
|
}
|
|
m_view->setWidgetVisible(this, false);
|
|
if ( m_widget ) {
|
|
m_widget->move(0, -500000);
|
|
m_widget->hide();
|
|
}
|
|
}
|
|
}
|
|
|
|
void RenderWidget::layout( )
|
|
{
|
|
KHTMLAssert( needsLayout() );
|
|
KHTMLAssert( minMaxKnown() );
|
|
if ( m_widget ) {
|
|
resizeWidget( m_width-borderLeft()-borderRight()-paddingLeft()-paddingRight(),
|
|
m_height-borderTop()-borderBottom()-paddingTop()-paddingBottom() );
|
|
if (!isRedirectedWidget() && (!isFrame() || document()->part()->parentPart()) && !m_needsMask) {
|
|
m_needsMask = true;
|
|
RenderLayer* rl = enclosingStackingContext();
|
|
RenderLayer* el = enclosingLayer();
|
|
while (rl && el && el != rl) {
|
|
if (el->renderer()->style()->position() != PSTATIC) {
|
|
m_needsMask = false;
|
|
break;
|
|
}
|
|
el = el->parent();
|
|
}
|
|
if (m_needsMask) {
|
|
if (rl) rl->setHasOverlaidWidgets();
|
|
canvas()->setNeedsWidgetMasks();
|
|
}
|
|
}
|
|
}
|
|
|
|
setNeedsLayout(false);
|
|
}
|
|
|
|
void RenderWidget::updateFromElement()
|
|
{
|
|
if (m_widget && !qobject_cast<KHTMLView*>(m_widget)) {
|
|
// Color:
|
|
QColor color = style()->color();
|
|
if (forceTransparentText())
|
|
color = Qt::transparent;
|
|
QColor backgroundColor = style()->backgroundColor();
|
|
|
|
if (!backgroundColor.isValid() && !style()->htmlHacks())
|
|
backgroundColor = Qt::transparent;
|
|
|
|
bool hasBackgroundImage = style()->hasBackgroundImage();
|
|
// check if we have to paint our background and let it show through the widget
|
|
bool trans = ( isRedirectedWidget() && !qobject_cast<KUrlRequester*>(m_widget) &&
|
|
(hasBackgroundImage || (style()->hasBackground() && shouldPaintCSSBorders())) );
|
|
|
|
QPalette pal(QApplication::palette(m_widget));
|
|
// We need a non-transparent version for widgets with popups (e.g. kcombobox). The popups must not let
|
|
// the background show through.
|
|
QPalette non_trans_pal = pal;
|
|
|
|
if (color.isValid() || backgroundColor.isValid() || trans) {
|
|
int contrast_ = KGlobalSettings::contrast();
|
|
int highlightVal = 100 + (2*contrast_+4)*16/10;
|
|
int lowlightVal = 100 + (2*contrast_+4)*10;
|
|
bool shouldChangeBgPal = true;
|
|
|
|
if (!backgroundColor.isValid())
|
|
backgroundColor = pal.color( widget()->backgroundRole() );
|
|
else
|
|
shouldChangeBgPal = !( (backgroundColor == colorForCSSValue(CSS_VAL_WINDOW)) ||
|
|
(backgroundColor == colorForCSSValue(CSS_VAL_BUTTONFACE)) );
|
|
if (shouldChangeBgPal || trans) {
|
|
pal.setColor(widget()->backgroundRole(), trans ? QColor(0,0,0,0) : backgroundColor);
|
|
for ( int i = 0; i < QPalette::NColorGroups; ++i ) {
|
|
if (shouldChangeBgPal) {
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Window, backgroundColor );
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Light, backgroundColor.light(highlightVal) );
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Dark, backgroundColor.dark(lowlightVal) );
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Mid, backgroundColor.dark(120) );
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Midlight, backgroundColor.light(110) );
|
|
}
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Button, trans ? QColor(0,0,0,0):backgroundColor );
|
|
pal.setColor( (QPalette::ColorGroup)i, QPalette::Base, trans ? QColor(0,0,0,0):backgroundColor );
|
|
}
|
|
}
|
|
|
|
if ( color.isValid() ) {
|
|
struct ColorSet {
|
|
QPalette::ColorGroup cg;
|
|
QPalette::ColorRole cr;
|
|
};
|
|
const struct ColorSet toSet [] = {
|
|
{ QPalette::Active, QPalette::WindowText },
|
|
{ QPalette::Active, QPalette::ButtonText },
|
|
{ QPalette::Active, QPalette::Text },
|
|
{ QPalette::Inactive, QPalette::WindowText },
|
|
{ QPalette::Inactive, QPalette::ButtonText },
|
|
{ QPalette::Inactive, QPalette::Text },
|
|
{ QPalette::NColorGroups, QPalette::NColorRoles },
|
|
};
|
|
const ColorSet *set = toSet;
|
|
while( set->cg != QPalette::NColorGroups ) {
|
|
pal.setColor( set->cg, set->cr, color );
|
|
non_trans_pal.setColor( set->cg, set->cr, color );
|
|
++set;
|
|
}
|
|
|
|
QColor disfg = color;
|
|
int h, s, v;
|
|
disfg.getHsv( &h, &s, &v );
|
|
if (v > 128)
|
|
// dark bg, light fg - need a darker disabled fg
|
|
disfg = disfg.dark(lowlightVal);
|
|
else if (v > 64)
|
|
// light bg, dark fg - need a lighter disabled fg - but only if !black
|
|
disfg = disfg.light(highlightVal);
|
|
else
|
|
// for really dark fg - use darkgray disabled fg,
|
|
// as ::light is pretty useless in this range
|
|
disfg = Qt::darkGray;
|
|
pal.setColor(QPalette::Disabled,QPalette::WindowText,disfg);
|
|
pal.setColor(QPalette::Disabled,QPalette::Text,disfg);
|
|
pal.setColor(QPalette::Disabled,QPalette::ButtonText,disfg);
|
|
non_trans_pal.setColor(QPalette::Disabled,QPalette::WindowText,disfg);
|
|
non_trans_pal.setColor(QPalette::Disabled,QPalette::Text,disfg);
|
|
non_trans_pal.setColor(QPalette::Disabled,QPalette::ButtonText,disfg);
|
|
}
|
|
}
|
|
|
|
if ( (qobject_cast<QCheckBox*>(m_widget) || qobject_cast<QRadioButton*>(m_widget)) &&
|
|
(backgroundColor == Qt::transparent && !hasBackgroundImage) ) {
|
|
m_widget->setPalette(non_trans_pal);
|
|
} else {
|
|
m_widget->setPalette(pal);
|
|
}
|
|
|
|
// Combobox's popup colors
|
|
if (qobject_cast<QComboBox*>(m_widget)) {
|
|
// Background
|
|
if (hasBackgroundImage) {
|
|
non_trans_pal = QApplication::palette();
|
|
}
|
|
else if (backgroundColor.isValid() && backgroundColor != Qt::transparent) {
|
|
non_trans_pal.setColor(QPalette::Base, backgroundColor);
|
|
}
|
|
// mmh great, there's no accessor for the popup...
|
|
QList<QWidget*>l = qFindChildren<QWidget *>(m_widget, QString());
|
|
foreach(QWidget* w, l) {
|
|
if (QAbstractScrollArea* lView = qobject_cast<QAbstractScrollArea*>(w)) {
|
|
// we have the listview, climb up to reach its container.
|
|
assert( w->parentWidget() != m_widget );
|
|
if (w->parentWidget()) {
|
|
w->parentWidget()->setPalette(non_trans_pal);
|
|
// Set system color for vertical scrollbar
|
|
lView->verticalScrollBar()->setPalette(QApplication::palette());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (QAbstractScrollArea* scrollView = qobject_cast<QAbstractScrollArea*>(m_widget)) {
|
|
// Border
|
|
if (QFrame* frame = qobject_cast<QFrame*>(m_widget)) {
|
|
if (shouldDisableNativeBorders()) {
|
|
if (frame->frameShape() != QFrame::NoFrame) {
|
|
m_nativeFrameShape = frame->frameShape();
|
|
frame->setFrameShape(QFrame::NoFrame);
|
|
}
|
|
}
|
|
else if (m_nativeFrameShape != QFrame::NoFrame) {
|
|
frame->setFrameShape(m_nativeFrameShape);
|
|
}
|
|
}
|
|
// Scrollbars color (ie css extension)
|
|
scrollView->horizontalScrollBar()->setPalette(style()->palette());
|
|
scrollView->verticalScrollBar()->setPalette(style()->palette());
|
|
}
|
|
else if (QLineEdit* lineEdit = qobject_cast<QLineEdit*>(m_widget)) {
|
|
lineEdit->setFrame(!shouldDisableNativeBorders());
|
|
}
|
|
}
|
|
|
|
RenderReplaced::updateFromElement();
|
|
}
|
|
|
|
void RenderWidget::paintBoxDecorations(PaintInfo& paintInfo, int _tx, int _ty)
|
|
{
|
|
QRect r = QRect(_tx, _ty, width(), height());
|
|
QRect cr = r.intersected(paintInfo.r);
|
|
|
|
if (qobject_cast<QAbstractScrollArea*>(m_widget) || (isRedirectedWidget() &&
|
|
(style()->hasBackgroundImage() || (style()->hasBackground() && shouldPaintCSSBorders()))))
|
|
{
|
|
paintAllBackgrounds(paintInfo.p, style()->backgroundColor(), style()->backgroundLayers(),
|
|
cr, r.x(), r.y(), r.width(), r.height());
|
|
}
|
|
|
|
if (shouldPaintCSSBorders() && style()->hasBorder())
|
|
{
|
|
paintBorder(paintInfo.p, _tx, _ty, width(), height(), style());
|
|
}
|
|
}
|
|
|
|
void RenderWidget::slotWidgetDestructed()
|
|
{
|
|
if (m_view)
|
|
m_view->setWidgetVisible(this, false);
|
|
m_widget = 0;
|
|
}
|
|
|
|
void RenderWidget::setStyle(RenderStyle *_style)
|
|
{
|
|
RenderReplaced::setStyle(_style);
|
|
if(m_widget)
|
|
{
|
|
m_widget->setFont(style()->font());
|
|
if (style()->visibility() != VISIBLE) {
|
|
if (m_view)
|
|
m_view->setWidgetVisible(this, false);
|
|
m_widget->hide();
|
|
}
|
|
}
|
|
}
|
|
|
|
void RenderWidget::paint(PaintInfo& paintInfo, int _tx, int _ty)
|
|
{
|
|
|
|
// not visible or not even once layouted
|
|
if (style()->visibility() != VISIBLE || m_y <= -500000)
|
|
return;
|
|
|
|
_tx += m_x;
|
|
_ty += m_y;
|
|
|
|
int os = maximalOutlineSize(paintInfo.phase);
|
|
if ( (_ty - os > paintInfo.r.bottom()) || (_ty + m_height +os <= paintInfo.r.top()) ||
|
|
(_tx + m_width +os <= paintInfo.r.left()) || (_tx - os > paintInfo.r.right()) )
|
|
return;
|
|
|
|
if ((paintInfo.phase == PaintActionChildBackground || paintInfo.phase == PaintActionChildBackgrounds) &&
|
|
shouldPaintBackgroundOrBorder() && !qobject_cast<KUrlRequester*>(m_widget))
|
|
paintBoxDecorations(paintInfo, _tx, _ty);
|
|
|
|
if (paintInfo.phase == PaintActionOutline && style()->outlineWidth())
|
|
paintOutline(paintInfo.p, _tx, _ty, width(), height(), style());
|
|
|
|
if (!m_widget || !m_view || paintInfo.phase != PaintActionForeground)
|
|
return;
|
|
|
|
int xPos = _tx+borderLeft()+paddingLeft();
|
|
int yPos = _ty+borderTop()+paddingTop();
|
|
|
|
bool khtmlw = isRedirectedWidget();
|
|
int childw = m_widget->width();
|
|
int childh = m_widget->height();
|
|
if ( (childw == 2000 || childh == 3072) && m_widget->inherits( "KHTMLView" ) ) {
|
|
KHTMLView *vw = static_cast<KHTMLView *>(m_widget);
|
|
int cy = m_view->contentsY();
|
|
int ch = m_view->visibleHeight();
|
|
|
|
|
|
int childx = m_widget->pos().x();
|
|
int childy = m_widget->pos().y();
|
|
|
|
int xNew = xPos;
|
|
int yNew = childy;
|
|
|
|
// qDebug("cy=%d, ch=%d, childy=%d, childh=%d", cy, ch, childy, childh );
|
|
if ( childh == 3072 ) {
|
|
if ( cy + ch > childy + childh ) {
|
|
yNew = cy + ( ch - childh )/2;
|
|
} else if ( cy < childy ) {
|
|
yNew = cy + ( ch - childh )/2;
|
|
}
|
|
// qDebug("calculated yNew=%d", yNew);
|
|
}
|
|
yNew = qMin( yNew, yPos + m_height - childh );
|
|
yNew = qMax( yNew, yPos );
|
|
if ( yNew != childy || xNew != childx ) {
|
|
if ( vw->contentsHeight() < yNew - yPos + childh )
|
|
vw->resizeContents( vw->contentsWidth(), yNew - yPos + childh );
|
|
vw->setContentsPos( xNew - xPos, yNew - yPos );
|
|
}
|
|
xPos = xNew;
|
|
yPos = yNew;
|
|
}
|
|
m_view->setWidgetVisible(this, true);
|
|
if (!khtmlw)
|
|
m_view->addChild( m_widget, xPos, yPos );
|
|
else
|
|
m_view->addChild( m_widget, xPos, -500000 +yPos);
|
|
m_widget->show();
|
|
if (khtmlw) {
|
|
if ( KHTMLView* v = qobject_cast<KHTMLView*>(m_widget) ) {
|
|
// our buffers are dedicated to scrollbars.
|
|
if (v->verticalScrollBar()->isVisible() && (!m_buffer[0] || v->verticalScrollBar()->size() != m_buffer[0]->size())) {
|
|
delete m_buffer[0];
|
|
m_buffer[0] = new QPixmap( v->verticalScrollBar()->size() );
|
|
}
|
|
if (v->horizontalScrollBar()->isVisible() && (!m_buffer[1] || v->horizontalScrollBar()->size() != m_buffer[1]->size())) {
|
|
delete m_buffer[1];
|
|
m_buffer[1] = new QPixmap( v->horizontalScrollBar()->size() );
|
|
}
|
|
} else if (!m_buffer[0] || (m_widget->size() != m_buffer[0]->size())) {
|
|
assert(!m_buffer[1]);
|
|
delete m_buffer[0];
|
|
m_buffer[0] = new QPixmap( m_widget->size() );
|
|
}
|
|
paintWidget(paintInfo, m_widget, xPos, yPos, m_buffer);
|
|
}
|
|
}
|
|
|
|
static void setInPaintEventFlag(QWidget* w, bool b = true, bool recurse=true)
|
|
{
|
|
w->setAttribute(Qt::WA_WState_InPaintEvent, b);
|
|
|
|
if (!recurse)
|
|
return;
|
|
if (qobject_cast<KHTMLView*>(w)) {
|
|
setInPaintEventFlag(static_cast<KHTMLView*>(w)->widget(), b, false);
|
|
setInPaintEventFlag(static_cast<KHTMLView*>(w)->horizontalScrollBar(), b, false);
|
|
setInPaintEventFlag(static_cast<KHTMLView*>(w)->verticalScrollBar(), b, false);
|
|
return;
|
|
}
|
|
|
|
foreach(QObject* o, w->children()) {
|
|
QWidget* const cw = static_cast<QWidget*>(o);
|
|
if (o->isWidgetType() && ! cw->isWindow()
|
|
&& !(cw->windowModality() & Qt::ApplicationModal)) {
|
|
setInPaintEventFlag(cw, b);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void copyWidget(const QRect& r, QPainter *p, QWidget *widget, int tx, int ty, bool buffered = false, QPixmap* buffer = 0)
|
|
{
|
|
if (r.isNull() || r.isEmpty() )
|
|
return;
|
|
|
|
QPoint thePoint(tx, ty);
|
|
QTransform t = p->worldTransform();
|
|
|
|
if (!buffered && t.isTranslating()) {
|
|
thePoint.setX( thePoint.x()+ static_cast<int>(t.dx()) );
|
|
thePoint.setY( thePoint.y()+ static_cast<int>(t.dy()) );
|
|
}
|
|
QPixmap* pm = 0;
|
|
QPaintDevice *d = p->device();
|
|
#ifdef IN_PLACE_WIDGET_PAINTING
|
|
QPaintDevice *x = d;
|
|
bool vte = p->viewTransformEnabled();
|
|
bool wme = p->worldMatrixEnabled();
|
|
QRect w = p->window();
|
|
QRect v = p->viewport();
|
|
QRegion rg = p->clipRegion();
|
|
qreal op = p->opacity();
|
|
QPen pen = p->pen();
|
|
QBrush brush = p->brush();
|
|
QPainter::RenderHints rh = p->renderHints();
|
|
#endif
|
|
if (buffered) {
|
|
if (!widget->size().isValid())
|
|
return;
|
|
// TT says Qt 4's widget painting hits an NVidia RenderAccel bug/shortcoming
|
|
// resulting in pixmap buffers being unsuitable for reuse by more than one widget.
|
|
//
|
|
// Until a turnaround exist in Qt, we can't reliably use shared buffers.
|
|
// ### pm = PaintBuffer::grab(widget->size());
|
|
assert( buffer );
|
|
pm = buffer;
|
|
if (!pm->hasAlphaChannel()) {
|
|
pm->fill(Qt::transparent);
|
|
} else {
|
|
QPainter pp(pm);
|
|
pp.setCompositionMode( QPainter::CompositionMode_Source );
|
|
pp.fillRect(r, Qt::transparent);
|
|
}
|
|
d = pm;
|
|
} else {
|
|
p->end();
|
|
}
|
|
|
|
setInPaintEventFlag( widget, false );
|
|
|
|
widget->render( d, (buffered ? QPoint(0,0) : thePoint) + r.topLeft(), r);
|
|
|
|
setInPaintEventFlag( widget );
|
|
|
|
if (!buffered) {
|
|
#ifdef IN_PLACE_WIDGET_PAINTING
|
|
p->begin(x);
|
|
p->setWorldTransform(t);
|
|
p->setWindow(w);
|
|
p->setViewport(v);
|
|
p->setViewTransformEnabled( vte );
|
|
p->setWorldMatrixEnabled( wme );
|
|
p->setRenderHints( rh );
|
|
if (!rg.isEmpty())
|
|
p->setClipRegion(rg);
|
|
if (op < 1.0f)
|
|
p->setOpacity(op);
|
|
p->setPen(pen);
|
|
p->setBrush(brush);
|
|
#else
|
|
assert(false);
|
|
#endif
|
|
} else {
|
|
// transfer results
|
|
QPoint off(r.x(), r.y());
|
|
p->drawPixmap(thePoint+off, static_cast<QPixmap&>(*d), r);
|
|
// ### PaintBuffer::release(pm);
|
|
}
|
|
}
|
|
|
|
void RenderWidget::paintWidget(PaintInfo& pI, QWidget *widget, int tx, int ty, QPixmap* buffer[])
|
|
{
|
|
QPainter* const p = pI.p;
|
|
allowWidgetPaintEvents = true;
|
|
|
|
bool buffered =
|
|
#ifdef IN_PLACE_WIDGET_PAINTING
|
|
p->combinedMatrix().m22() != 1.0 || (p->device()->devType() == QInternal::Printer);
|
|
#else
|
|
true;
|
|
#endif
|
|
QRect rr = pI.r;
|
|
rr.translate(-tx, -ty);
|
|
const QRect r = widget->rect().intersect( rr );
|
|
if ( KHTMLView* v = qobject_cast<KHTMLView*>( widget ) ) {
|
|
QPoint thePoint(tx, ty);
|
|
if (v->verticalScrollBar()->isVisible()) {
|
|
QRect vbr = v->verticalScrollBar()->rect();
|
|
QPoint of = v->verticalScrollBar()->mapTo(v, QPoint(vbr.x(), vbr.y()));
|
|
vbr.translate( of );
|
|
vbr &= r;
|
|
vbr.translate( -of );
|
|
if (vbr.isValid() && !vbr.isEmpty())
|
|
copyWidget(vbr, p, v->verticalScrollBar(), tx+of.x(), ty+of.y(), buffered, buffer[0]);
|
|
}
|
|
if (v->horizontalScrollBar()->isVisible()) {
|
|
QRect hbr = v->horizontalScrollBar()->rect();
|
|
QPoint of = v->horizontalScrollBar()->mapTo(v, QPoint(hbr.x(), hbr.y()));
|
|
hbr.translate( of );
|
|
hbr &= r;
|
|
hbr.translate( -of );
|
|
if (hbr.isValid() && !hbr.isEmpty())
|
|
copyWidget(hbr, p, v->horizontalScrollBar(), tx+ of.x(), ty+ of.y(), buffered, buffer[1]);
|
|
}
|
|
QPoint of = v->viewport()->mapTo(v, QPoint(0,0));
|
|
QRect vr = (r & v->viewport()->rect().translated(of));
|
|
if (vr.isValid() && !vr.isEmpty())
|
|
v->render(p, vr, thePoint);
|
|
} else {
|
|
copyWidget(r, p, widget, tx, ty, buffered, buffer[0]);
|
|
}
|
|
allowWidgetPaintEvents = false;
|
|
}
|
|
|
|
bool RenderWidget::eventFilter(QObject* /*o*/, QEvent* e)
|
|
{
|
|
// no special event processing if this is a frame (in which case KHTMLView handles it all)
|
|
if ( qobject_cast<KHTMLView*>( m_widget ) || isRedirectedWidget() )
|
|
return false;
|
|
if ( !element() ) return true;
|
|
|
|
|
|
static bool directToWidget = false;
|
|
if (directToWidget) {
|
|
//We're trying to get the event to the widget
|
|
//promptly. So get out of here..
|
|
return false;
|
|
}
|
|
|
|
ref();
|
|
element()->ref();
|
|
|
|
bool filtered = false;
|
|
|
|
//kDebug() << "RenderWidget::eventFilter type=" << e->type();
|
|
switch(e->type()) {
|
|
case QEvent::FocusOut:
|
|
// First, forward it to the widget, so that Qt gets a precise
|
|
// state of the focus before pesky JS can try changing it..
|
|
directToWidget = true;
|
|
QApplication::sendEvent(m_widget, e);
|
|
directToWidget = false;
|
|
filtered = true; //We already delivered it!
|
|
|
|
// Don't count popup as a valid reason for losing the focus
|
|
// (example: opening the options of a select combobox shouldn't emit onblur)
|
|
if ( static_cast<QFocusEvent*>(e)->reason() != Qt::PopupFocusReason )
|
|
handleFocusOut();
|
|
break;
|
|
case QEvent::FocusIn:
|
|
//As above, forward to the widget first...
|
|
directToWidget = true;
|
|
QApplication::sendEvent(m_widget, e);
|
|
directToWidget = false;
|
|
filtered = true; //We already delivered it!
|
|
|
|
//kDebug(6000) << "RenderWidget::eventFilter captures FocusIn";
|
|
document()->setFocusNode(element());
|
|
// if ( isEditable() ) {
|
|
// KHTMLPartBrowserExtension *ext = static_cast<KHTMLPartBrowserExtension *>( element()->view->part()->browserExtension() );
|
|
// if ( ext ) ext->editableWidgetFocused( m_widget );
|
|
// }
|
|
break;
|
|
case QEvent::Wheel: {
|
|
if (widget()->parentWidget() == view()->widget()) {
|
|
bool vertical = ( static_cast<QWheelEvent*>(e)->orientation() == Qt::Vertical );
|
|
// don't allow the widget to react to wheel event if
|
|
// the view is being scrolled by mouse wheel
|
|
// This does not apply if the webpage has no valid scroll range in the given wheel event orientation.
|
|
if ( ((vertical && (view()->contentsHeight() > view()->visibleHeight())) ||
|
|
(!vertical && (view()->contentsWidth() > view()->visibleWidth()))) &&
|
|
view()->isScrollingFromMouseWheel() ) {
|
|
static_cast<QWheelEvent*>(e)->ignore();
|
|
QApplication::sendEvent(view(), e);
|
|
filtered = true;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case QEvent::KeyPress:
|
|
case QEvent::KeyRelease:
|
|
// TODO this seems wrong - Qt events are not correctly translated to DOM ones,
|
|
// like in KHTMLView::dispatchKeyEvent()
|
|
if (element()->dispatchKeyEvent(static_cast<QKeyEvent*>(e),false))
|
|
filtered = true;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
};
|
|
|
|
element()->deref();
|
|
|
|
// stop processing if the widget gets deleted, but continue in all other cases
|
|
if (hasOneRef())
|
|
filtered = true;
|
|
deref();
|
|
|
|
return filtered;
|
|
}
|
|
|
|
void RenderWidget::EventPropagator::sendEvent(QEvent *e) {
|
|
// ### why don't we just call event()? That would be the normal route.
|
|
switch(e->type()) {
|
|
case QEvent::Wheel:
|
|
wheelEvent ( static_cast<QWheelEvent *> (e) );
|
|
break;
|
|
case QEvent::MouseButtonPress:
|
|
mousePressEvent(static_cast<QMouseEvent *>(e));
|
|
break;
|
|
case QEvent::MouseButtonRelease:
|
|
mouseReleaseEvent(static_cast<QMouseEvent *>(e));
|
|
break;
|
|
case QEvent::MouseButtonDblClick:
|
|
mouseDoubleClickEvent(static_cast<QMouseEvent *>(e));
|
|
break;
|
|
case QEvent::MouseMove:
|
|
mouseMoveEvent(static_cast<QMouseEvent *>(e));
|
|
break;
|
|
case QEvent::KeyPress:
|
|
keyPressEvent(static_cast<QKeyEvent *>(e));
|
|
break;
|
|
case QEvent::KeyRelease:
|
|
if (qobject_cast<QLineEdit*>(this))
|
|
event(e);
|
|
else
|
|
keyReleaseEvent(static_cast<QKeyEvent *>(e));
|
|
break;
|
|
case QEvent::FocusIn:
|
|
focusInEvent(static_cast<QFocusEvent *>(e));
|
|
break;
|
|
case QEvent::FocusOut:
|
|
focusOutEvent(static_cast<QFocusEvent *>(e));
|
|
break;
|
|
case QEvent::ContextMenu:
|
|
contextMenuEvent(static_cast<QContextMenuEvent*>(e));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// send event to target and let it bubble up until it is accepted or
|
|
// once it would go through a limiting widget level
|
|
static bool bubblingSend(QWidget* target, QEvent* e, QWidget* stoppingParent)
|
|
{
|
|
for (;;) {
|
|
static_cast<RenderWidget::EventPropagator *>(target)->sendEvent(e);
|
|
if (e->isAccepted())
|
|
return true;
|
|
if (target == stoppingParent)
|
|
return false;
|
|
// ### might want to shift Q*Event::pos() as we go up
|
|
target = target->parentWidget();
|
|
assert(target != 0);
|
|
}
|
|
}
|
|
|
|
bool RenderWidget::handleEvent(const DOM::EventImpl& ev)
|
|
{
|
|
bool ret = false;
|
|
|
|
switch(ev.id()) {
|
|
case EventImpl::DOMFOCUSIN_EVENT:
|
|
case EventImpl::DOMFOCUSOUT_EVENT: {
|
|
QFocusEvent e(ev.id() == EventImpl::DOMFOCUSIN_EVENT ? QEvent::FocusIn : QEvent::FocusOut);
|
|
// E.g. a KLineEdit child widget might be defined to receive
|
|
// focus instead
|
|
QWidget* fw = m_widget->focusProxy() ? m_widget->focusProxy() : m_widget;
|
|
static_cast<EventPropagator *>(fw)->sendEvent(&e);
|
|
ret = e.isAccepted();
|
|
break;
|
|
}
|
|
case EventImpl::KHTML_MOUSEWHEEL_EVENT:
|
|
case EventImpl::MOUSEDOWN_EVENT:
|
|
case EventImpl::MOUSEUP_EVENT:
|
|
case EventImpl::MOUSEMOVE_EVENT: {
|
|
if (!ev.isMouseEvent()) break;
|
|
|
|
const MouseEventImpl &me = static_cast<const MouseEventImpl &>(ev);
|
|
QMouseEvent* const qme = me.qEvent();
|
|
QMouseEvent::Type type;
|
|
Qt::MouseButton button = Qt::NoButton;
|
|
Qt::MouseButtons buttons = Qt::NoButton;
|
|
Qt::KeyboardModifiers state = 0;
|
|
Qt::Orientation orient = Qt::Vertical;
|
|
|
|
if (qme) {
|
|
buttons = qme->buttons();
|
|
button = qme->button();
|
|
state = qme->modifiers();
|
|
type = qme->type();
|
|
} else {
|
|
switch (me.button()) {
|
|
case 0:
|
|
button = Qt::LeftButton;
|
|
break;
|
|
case 1:
|
|
button = Qt::MidButton;
|
|
break;
|
|
case 2:
|
|
button = Qt::RightButton;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
buttons = button;
|
|
|
|
switch(me.id()) {
|
|
case EventImpl::MOUSEDOWN_EVENT:
|
|
type = QMouseEvent::MouseButtonPress;
|
|
break;
|
|
case EventImpl::MOUSEUP_EVENT:
|
|
type = QMouseEvent::MouseButtonRelease;
|
|
break;
|
|
case EventImpl::KHTML_MOUSEWHEEL_EVENT:
|
|
case EventImpl::MOUSEMOVE_EVENT:
|
|
default:
|
|
type = QMouseEvent::MouseMove;
|
|
button = Qt::NoButton;
|
|
break;
|
|
}
|
|
|
|
if (me.orientation() == MouseEventImpl::OHorizontal)
|
|
orient = Qt::Horizontal;
|
|
|
|
if (me.ctrlKey())
|
|
state |= Qt::ControlModifier;
|
|
if (me.altKey())
|
|
state |= Qt::AltModifier;
|
|
if (me.shiftKey())
|
|
state |= Qt::ShiftModifier;
|
|
if (me.metaKey())
|
|
state |= Qt::MetaModifier;
|
|
}
|
|
|
|
int absx = 0;
|
|
int absy = 0;
|
|
absolutePosition(absx, absy);
|
|
absx += borderLeft()+paddingLeft();
|
|
absy += borderTop()+paddingTop();
|
|
|
|
QPoint p(me.clientX() - absx + m_view->contentsX(),
|
|
me.clientY() - absy + m_view->contentsY());
|
|
|
|
if (ev.id() == EventImpl::MOUSEMOVE_EVENT && view()->mouseEventsTarget()) {
|
|
// mitigate the problem that mouse moves outside of the widget rect
|
|
// interrupt the selection process in textarea (#156574)
|
|
// e.g. when extending a selection upward to make a textarea scroll
|
|
// while selecting its content.
|
|
// We don't emulate properly what Qt does it seems,tough I verified
|
|
// the move event is properly forwarded to the textarea and has
|
|
// appropriate coordinates. Might be Enter/Leave issue. ### FIXME
|
|
// In the meantime, clamping the event to the widget rect
|
|
// will at least prevent the selection to be lost.
|
|
p.setX(qMin(qMax(0,p.x()),m_widget->width()));
|
|
p.setY(qMin(qMax(0,p.y()),m_widget->height()));
|
|
}
|
|
|
|
QPointer<QWidget> target;
|
|
target = m_widget->childAt(p);
|
|
|
|
if (target) {
|
|
p = target->mapFrom(m_widget, p);
|
|
}
|
|
|
|
if (m_underMouse != target && ev.id() != EventImpl::KHTML_MOUSEWHEEL_EVENT) {
|
|
if (m_underMouse) {
|
|
QEvent moe( QEvent::Leave );
|
|
QApplication::sendEvent(m_underMouse, &moe);
|
|
// qDebug() << "sending LEAVE to"<< m_underMouse;
|
|
if (m_underMouse->testAttribute(Qt::WA_Hover)) {
|
|
QHoverEvent he( QEvent::HoverLeave, QPoint(-1,-1), QPoint(0,0) );
|
|
QApplication::sendEvent(m_underMouse, &he);
|
|
}
|
|
}
|
|
if (target) {
|
|
QEvent moe( QEvent::Enter );
|
|
QApplication::sendEvent(target, &moe);
|
|
// qDebug() << "sending ENTER to" << target;
|
|
if (target->testAttribute(Qt::WA_Hover)) {
|
|
QHoverEvent he( QEvent::HoverEnter, QPoint(0,0), QPoint(-1,-1) );
|
|
QApplication::sendEvent(target, &he);
|
|
}
|
|
}
|
|
m_underMouse = target;
|
|
}
|
|
#if 0
|
|
if (target && ev.id() == EventImpl::MOUSEMOVE_EVENT) {
|
|
// ### is this one still necessary? it doubles every mouse event...
|
|
// I'd reckon it's no longer needed since Harri made the event propagation bubble
|
|
QMouseEvent evt(QEvent::MouseMove, p, Qt::NoButton,
|
|
QApplication::mouseButtons(), QApplication::keyboardModifiers());
|
|
QApplication::sendEvent(target, &evt);
|
|
}
|
|
#endif
|
|
if (ev.id() == EventImpl::MOUSEDOWN_EVENT) {
|
|
if (!target || (!::qobject_cast<QScrollBar*>(target) &&
|
|
!::qobject_cast<KUrlRequester*>(m_widget) &&
|
|
!::qobject_cast<QLineEdit*>(m_widget)))
|
|
target = m_widget;
|
|
view()->setMouseEventsTarget( target );
|
|
} else {
|
|
target = view()->mouseEventsTarget();
|
|
if (target) {
|
|
QWidget * parent = target;
|
|
while (parent && parent != m_widget)
|
|
parent = parent->parentWidget();
|
|
if (!parent) return false;
|
|
} else {
|
|
target = m_widget;
|
|
}
|
|
}
|
|
|
|
bool needContextMenuEvent = (type == QMouseEvent::MouseButtonPress && button == Qt::RightButton);
|
|
bool isMouseWheel = (ev.id() == EventImpl::KHTML_MOUSEWHEEL_EVENT);
|
|
|
|
if (isMouseWheel) {
|
|
// don't allow the widget to react to wheel event if
|
|
// a) the view is being scrolled by mouse wheel
|
|
// b) it's an unfocused ComboBox (for extra security against unwanted changes to formulars)
|
|
// This does not apply if the webpage has no valid scroll range in the given wheel event orientation.
|
|
if ( ((orient == Qt::Vertical && (view()->contentsHeight() > view()->visibleHeight())) ||
|
|
(orient == Qt::Horizontal && (view()->contentsWidth() > view()->visibleWidth()))) &&
|
|
( view()->isScrollingFromMouseWheel() ||
|
|
(qobject_cast<QComboBox*>(m_widget) &&
|
|
(!document()->focusNode() || document()->focusNode()->renderer() != this) ))) {
|
|
ret = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
QScopedPointer<QEvent> e(isMouseWheel ?
|
|
static_cast<QEvent*>(new QWheelEvent(p, -me.detail()*40, buttons, state, orient)) :
|
|
static_cast<QEvent*>(new QMouseEvent(type, p, button, buttons, state)));
|
|
|
|
|
|
ret = bubblingSend(target, e.data(), m_widget);
|
|
|
|
if (!target)
|
|
break;
|
|
if (needContextMenuEvent) {
|
|
QContextMenuEvent cme(QContextMenuEvent::Mouse, p);
|
|
static_cast<EventPropagator *>(target.data())->sendEvent(&cme);
|
|
} else if (type == QEvent::MouseMove && target->testAttribute(Qt::WA_Hover)) {
|
|
QHoverEvent he( QEvent::HoverMove, p, p );
|
|
QApplication::sendEvent(target, &he);
|
|
}
|
|
if (ev.id() == EventImpl::MOUSEUP_EVENT) {
|
|
view()->setMouseEventsTarget( 0 );
|
|
}
|
|
break;
|
|
}
|
|
case EventImpl::KEYDOWN_EVENT:
|
|
// do nothing; see the mapping table below
|
|
break;
|
|
case EventImpl::KEYUP_EVENT: {
|
|
if (!ev.isKeyRelatedEvent()) break;
|
|
|
|
const KeyEventBaseImpl& domKeyEv = static_cast<const KeyEventBaseImpl &>(ev);
|
|
if (domKeyEv.isSynthetic() && !acceptsSyntheticEvents()) break;
|
|
|
|
QKeyEvent* const ke = domKeyEv.qKeyEvent();
|
|
static_cast<EventPropagator *>(m_widget)->sendEvent(ke);
|
|
ret = ke->isAccepted();
|
|
break;
|
|
}
|
|
case EventImpl::KEYPRESS_EVENT: {
|
|
if (!ev.isKeyRelatedEvent()) break;
|
|
|
|
const KeyEventBaseImpl& domKeyEv = static_cast<const KeyEventBaseImpl &>(ev);
|
|
if (domKeyEv.isSynthetic() && !acceptsSyntheticEvents()) break;
|
|
|
|
// See KHTMLView::dispatchKeyEvent: autorepeat is just keypress in the DOM
|
|
// but it's keyrelease+keypress in Qt. So here we do the inverse mapping as
|
|
// the one done in KHTMLView: generate two events for one DOM auto-repeat keypress.
|
|
// Similarly, DOM keypress events with non-autorepeat Qt event do nothing here,
|
|
// because the matching Qt keypress event was already sent from DOM keydown event.
|
|
|
|
// Reverse drawing as the one in KHTMLView:
|
|
// DOM: Down Press | Press | Up
|
|
// Qt: (nothing) Press | Release(autorepeat) + Press(autorepeat) | Release
|
|
//
|
|
// Qt::KeyPress is sent for DOM keypress and not DOM keydown to allow
|
|
// sites to block a key with onkeypress, #99749
|
|
|
|
QKeyEvent* const ke = domKeyEv.qKeyEvent();
|
|
if (ke->isAutoRepeat()) {
|
|
QKeyEvent releaseEv( QEvent::KeyRelease, ke->key(), ke->modifiers(),
|
|
ke->text(), ke->isAutoRepeat(), ke->count() );
|
|
static_cast<EventPropagator *>(m_widget)->sendEvent(&releaseEv);
|
|
}
|
|
QWidget *fw = m_widget->focusWidget() ? m_widget->focusWidget() : m_widget;
|
|
static_cast<EventPropagator *>(fw)->sendEvent(ke);
|
|
ret = ke->isAccepted();
|
|
break;
|
|
}
|
|
case EventImpl::MOUSEOUT_EVENT: {
|
|
QWidget* target = m_underMouse ? (QWidget*) m_underMouse : m_widget;
|
|
QEvent moe( QEvent::Leave );
|
|
QApplication::sendEvent(target, &moe);
|
|
// qDebug() << "received MOUSEOUT, forwarding to" << target ;
|
|
if (target->testAttribute(Qt::WA_Hover)) {
|
|
QHoverEvent he( QEvent::HoverLeave, QPoint(-1,-1), QPoint(0,0) );
|
|
QApplication::sendEvent(target, &he);
|
|
}
|
|
m_underMouse = 0;
|
|
break;
|
|
}
|
|
case EventImpl::MOUSEOVER_EVENT: {
|
|
QEvent moe( QEvent::Enter );
|
|
QApplication::sendEvent(m_widget, &moe);
|
|
// qDebug() << "received MOUSEOVER, forwarding to" << m_widget;
|
|
if (m_widget->testAttribute(Qt::WA_Hover)) {
|
|
QHoverEvent he( QEvent::HoverEnter, QPoint(0,0), QPoint(-1,-1) );
|
|
QApplication::sendEvent(m_widget, &he);
|
|
}
|
|
view()->part()->resetHoverText();
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void RenderWidget::deref()
|
|
{
|
|
if (_ref) _ref--;
|
|
// qDebug( "deref(%p): width get count is %d", this, _ref);
|
|
if (!_ref) {
|
|
if (attached()) {
|
|
_ref++;
|
|
detach(); // will perform the final deref.
|
|
return;
|
|
}
|
|
SharedPtr<RenderArena> guard(m_arena); //Since delete on us gets called -first-,
|
|
//before the arena free
|
|
arenaDelete(m_arena.get());
|
|
}
|
|
}
|
|
|
|
#ifdef ENABLE_DUMP
|
|
void RenderWidget::dump(QTextStream &stream, const QString &ind) const
|
|
{
|
|
RenderReplaced::dump(stream,ind);
|
|
if ( widget() )
|
|
stream << " color=" << widget()->palette().color( widget()->foregroundRole() ).name()
|
|
<< " bg=" << widget()->palette().color( widget()->backgroundRole() ).name();
|
|
else
|
|
stream << " null widget";
|
|
}
|
|
#endif
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
QPoint KHTMLWidgetPrivate::absolutePos()
|
|
{
|
|
if (!m_rw)
|
|
return m_pos;
|
|
int x, y;
|
|
m_rw->absolutePosition(x, y);
|
|
x += m_rw->borderLeft()+m_rw->paddingLeft();
|
|
y += m_rw->borderTop()+m_rw->paddingTop();
|
|
return QPoint(x, y);
|
|
}
|
|
|
|
KHTMLView* KHTMLWidgetPrivate::rootViewPos(QPoint& pos)
|
|
{
|
|
if (!m_rw || !m_rw->widget()) {
|
|
pos = QPoint();
|
|
return 0;
|
|
}
|
|
pos = absolutePos();
|
|
KHTMLView* v = m_rw->view();
|
|
KHTMLView* last = 0;
|
|
while (v) {
|
|
last = v;
|
|
int w,h = 0;
|
|
v->applyTransforms(pos.rx(), pos.ry(), w, h);
|
|
KHTMLWidget*kw = dynamic_cast<KHTMLWidget*>(v);
|
|
if (!kw || !kw->m_kwp->isRedirected())
|
|
break;
|
|
pos += kw->m_kwp->absolutePos();
|
|
v = v->part()->parentPart() ? v->part()->parentPart()->view() : 0;
|
|
}
|
|
return last;
|
|
}
|
|
|
|
void KHTMLWidgetPrivate::setIsRedirected( bool b )
|
|
{
|
|
m_redirected = b;
|
|
if (!b && m_rw && m_rw->widget()) {
|
|
setInPaintEventFlag( m_rw->widget(), false );
|
|
m_rw->widget()->setAttribute(Qt::WA_OpaquePaintEvent, false);
|
|
m_rw->widget()->removeEventFilter(m_rw->view());
|
|
}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
KHTMLWidget::KHTMLWidget()
|
|
: m_kwp(new KHTMLWidgetPrivate()) {}
|
|
|
|
KHTMLWidget::~KHTMLWidget()
|
|
{ delete m_kwp; }
|
|
|
|
#include "render_replaced.moc"
|
|
|