You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdelibs/khtml/rendering/render_replaced.cpp

940 lines
30 KiB

/**
* This file is part of the HTML widget for KDE.
*
* Copyright (C) 1999-2003 Lars Knoll (knoll@kde.org)
* Copyright (C) 2000-2003 Dirk Mueller (mueller@kde.org)
* Copyright (C) 2003 Apple Computer, Inc.
* Copyright (C) 2004 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 "render_arena.h"
#include <assert.h>
#include <qwidget.h>
#include <qpainter.h>
#include <qevent.h>
#include <qapplication.h>
#include <qlineedit.h>
#include <kglobalsettings.h>
#include <qobjectlist.h>
#include <qvaluevector.h>
#include "khtml_ext.h"
#include "khtmlview.h"
#include "xml/dom2_eventsimpl.h"
#include "khtml_part.h"
#include "xml/dom_docimpl.h"
#include <kdebug.h>
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
kdDebug( 6040 ) << "RenderReplaced::calcMinMaxWidth() known=" << minMaxKnown() << endl;
#endif
m_width = calcReplacedWidth();
m_width = calcBoxWidth( m_width );
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();
}
void RenderReplaced::position(InlineBox* box, int /*from*/, int /*len*/, bool /*reverse*/)
{
setPos( box->xPos(), box->yPos() );
}
// -----------------------------------------------------------------------------
RenderWidget::RenderWidget(DOM::NodeImpl* node)
: RenderReplaced(node)
{
m_widget = 0;
// a widget doesn't support being anonymous
assert(!isAnonymous());
m_view = node->getDocument()->view();
m_arena.reset(renderArena());
m_resizePending = false;
m_discardResizes = false;
m_isKHTMLWidget = false;
m_needsMask = false;
// this is no real reference counting, its just there
// to make sure that we're not deleted while we're recursed
// in an eventFilter of the widget
ref();
}
void RenderWidget::detach()
{
remove();
deleteInlineBoxes();
if ( m_widget ) {
if ( m_view ) {
m_view->setWidgetVisible(this, false);
m_view->removeChild( m_widget );
}
m_widget->removeEventFilter( this );
m_widget->setMouseTracking( false );
}
deref();
}
RenderWidget::~RenderWidget()
{
KHTMLAssert( refCount() <= 0 );
if(m_widget) {
m_widget->hide();
m_widget->deleteLater();
}
}
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
// its bigger )
h = kMin( h, 3072 );
w = kMin( w, 2000 );
if (m_widget->width() != w || m_widget->height() != h) {
m_resizePending = isKHTMLWidget();
ref();
element()->ref();
QApplication::postEvent( this, new QWidgetResizeEvent( w, h ) );
element()->deref();
deref();
}
}
void RenderWidget::cancelPendingResize()
{
if (!m_widget)
return;
m_discardResizes = true;
QApplication::sendPostedEvents(this, QWidgetResizeEvent::Type);
m_discardResizes = false;
}
bool RenderWidget::event( QEvent *e )
{
if ( m_widget && (e->type() == (QEvent::Type)QWidgetResizeEvent::Type) ) {
m_resizePending = false;
if (m_discardResizes)
return true;
QWidgetResizeEvent *re = static_cast<QWidgetResizeEvent *>(e);
m_widget->resize( re->w, re->h );
repaint();
}
// eat all events - except if this is a frame (in which case KHTMLView handles it all)
if ( ::qt_cast<KHTMLView *>( m_widget ) )
return QObject::event( e );
return true;
}
void RenderWidget::flushWidgetResizes() //static
{
QApplication::sendPostedEvents( 0, QWidgetResizeEvent::Type );
}
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();
m_widget->deleteLater(); //Might happen due to event on the widget, so be careful
m_widget = 0;
}
m_widget = widget;
if (m_widget) {
connect( m_widget, SIGNAL( destroyed()), this, SLOT( slotWidgetDestructed()));
m_widget->installEventFilter(this);
if ( (m_isKHTMLWidget = !strcmp(m_widget->name(), "__khtml")) && !::qt_cast<QFrame*>(m_widget))
m_widget->setBackgroundMode( QWidget::NoBackground );
if (m_widget->focusPolicy() > QWidget::StrongFocus)
m_widget->setFocusPolicy(QWidget::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);
m_view->addChild( m_widget, 0, -500000);
if ( m_widget ) m_widget->hide();
m_resizePending = false;
}
}
void RenderWidget::layout( )
{
KHTMLAssert( needsLayout() );
KHTMLAssert( minMaxKnown() );
if ( m_widget ) {
resizeWidget( m_width-borderLeft()-borderRight()-paddingLeft()-paddingRight(),
m_height-borderTop()-borderBottom()-paddingTop()-paddingBottom() );
if (!isKHTMLWidget() && !isFrame() && !m_needsMask) {
m_needsMask = true;
RenderLayer* rl = enclosingStackingContext();
RenderLayer* el = enclosingLayer();
while (rl && el && el != rl) {
if (el->renderer()->style()->position() != STATIC) {
m_needsMask = false;
break;
}
el = el->parent();
}
if (m_needsMask) {
if (rl) rl->setHasOverlaidWidgets();
canvas()->setNeedsWidgetMasks();
}
}
}
setNeedsLayout(false);
}
void RenderWidget::updateFromElement()
{
if (m_widget) {
// Color:
QColor color = style()->color();
QColor backgroundColor = style()->backgroundColor();
if ( color.isValid() || backgroundColor.isValid() ) {
QPalette pal(QApplication::palette(m_widget));
int contrast_ = KGlobalSettings::contrast();
int highlightVal = 100 + (2*contrast_+4)*16/10;
int lowlightVal = 100 + (2*contrast_+4)*10;
if (backgroundColor.isValid()) {
if (!isKHTMLWidget())
widget()->setEraseColor(backgroundColor );
for ( int i = 0; i < QPalette::NColorGroups; ++i ) {
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Background, backgroundColor );
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Light, backgroundColor.light(highlightVal) );
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Dark, backgroundColor.dark(lowlightVal) );
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Mid, backgroundColor.dark(120) );
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Midlight, backgroundColor.light(110) );
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Button, backgroundColor );
pal.setColor( (QPalette::ColorGroup)i, QColorGroup::Base, backgroundColor );
}
}
if ( color.isValid() ) {
struct ColorSet {
QPalette::ColorGroup cg;
QColorGroup::ColorRole cr;
};
const struct ColorSet toSet [] = {
{ QPalette::Active, QColorGroup::Foreground },
{ QPalette::Active, QColorGroup::ButtonText },
{ QPalette::Active, QColorGroup::Text },
{ QPalette::Inactive, QColorGroup::Foreground },
{ QPalette::Inactive, QColorGroup::ButtonText },
{ QPalette::Inactive, QColorGroup::Text },
{ QPalette::Disabled,QColorGroup::ButtonText },
{ QPalette::NColorGroups, QColorGroup::NColorRoles },
};
const ColorSet *set = toSet;
while( set->cg != QPalette::NColorGroups ) {
pal.setColor( set->cg, set->cr, color );
++set;
}
QColor disfg = color;
int h, s, v;
disfg.hsv( &h, &s, &v );
if (v > 128)
// dark bg, light fg - need a darker disabled fg
disfg = disfg.dark(lowlightVal);
else if (disfg != Qt::black)
// light bg, dark fg - need a lighter disabled fg - but only if !black
disfg = disfg.light(highlightVal);
else
// black fg - use darkgray disabled fg
disfg = Qt::darkGray;
pal.setColor(QPalette::Disabled,QColorGroup::Foreground,disfg);
}
m_widget->setPalette(pal);
}
else
m_widget->unsetPalette();
// Border:
QFrame* frame = ::qt_cast<QFrame*>(m_widget);
if (frame) {
if (shouldPaintBackgroundOrBorder())
{
frame->setFrameShape(QFrame::NoFrame);
}
}
}
RenderReplaced::updateFromElement();
}
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();
}
}
// Don't paint borders if the border-style is native
// or borders are not supported on this widget
if (!canHaveBorder() ||
(style()->borderLeftStyle() == BNATIVE &&
style()->borderRightStyle() == BNATIVE &&
style()->borderTopStyle() == BNATIVE &&
style()->borderBottomStyle() == BNATIVE))
{
setShouldPaintBackgroundOrBorder(false);
}
}
void RenderWidget::paint(PaintInfo& paintInfo, int _tx, int _ty)
{
_tx += m_x;
_ty += m_y;
if (shouldPaintBackgroundOrBorder() &&
(paintInfo.phase == PaintActionChildBackground || paintInfo.phase == PaintActionChildBackgrounds))
paintBoxDecorations(paintInfo, _tx, _ty);
if (!m_widget || !m_view || paintInfo.phase != PaintActionForeground)
return;
// not visible or not even once layouted
if (style()->visibility() != VISIBLE || m_y <= -500000 || m_resizePending )
return;
if ( (_ty > paintInfo.r.bottom()) || (_ty + m_height <= paintInfo.r.top()) ||
(_tx + m_width <= paintInfo.r.left()) || (_tx > paintInfo.r.right()) )
return;
int xPos = _tx+borderLeft()+paddingLeft();
int yPos = _ty+borderTop()+paddingTop();
bool khtmlw = isKHTMLWidget();
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_view->childX( m_widget );
int childy = m_view->childY( m_widget );
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 = kMin( yNew, yPos + m_height - childh );
yNew = kMax( 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);
m_view->addChild(m_widget, xPos, yPos );
m_widget->show();
if (khtmlw)
paintWidget(paintInfo, m_widget, xPos, yPos);
}
#include <private/qinternal_p.h>
// The PaintBuffer class provides a shared buffer for widget painting.
//
// It will grow to encompass the biggest widget encountered, in order to avoid
// constantly resizing.
// When it grows over maxPixelBuffering, it periodically checks if such a size
// is still needed. If not, it shrinks down to the biggest size < maxPixelBuffering
// that was requested during the overflow lapse.
class PaintBuffer: public QObject
{
public:
static const int maxPixelBuffering = 320*200;
static const int leaseTime = 20*1000;
static QPixmap *grab( QSize s = QSize() ) {
if (!m_inst)
m_inst = new PaintBuffer;
return m_inst->getBuf( s );
}
static void release() { m_inst->m_grabbed = false; }
protected:
PaintBuffer(): m_overflow(false), m_grabbed(false),
m_timer(0), m_resetWidth(0), m_resetHeight(0) {};
void timerEvent(QTimerEvent* e) {
assert( m_timer == e->timerId() );
if (m_grabbed)
return;
m_buf.resize(m_resetWidth, m_resetHeight);
m_resetWidth = m_resetHeight = 0;
killTimer( m_timer );
m_timer = 0;
}
QPixmap *getBuf( QSize s ) {
assert( !m_grabbed );
if (s.isEmpty())
return 0;
m_grabbed = true;
bool cur_overflow = false;
int nw = kMax(m_buf.width(), s.width());
int nh = kMax(m_buf.height(), s.height());
if (!m_overflow && (nw*nh > maxPixelBuffering))
cur_overflow = true;
if (nw != m_buf.width() || nh != m_buf.height())
m_buf.resize(nw, nh);
if (cur_overflow) {
m_overflow = true;
m_timer = startTimer( leaseTime );
} else if (m_overflow) {
if( s.width()*s.height() > maxPixelBuffering ) {
killTimer( m_timer );
m_timer = startTimer( leaseTime );
} else {
if (s.width() > m_resetWidth)
m_resetWidth = s.width();
if (s.height() > m_resetHeight)
m_resetHeight = s.height();
}
}
return &m_buf;
}
private:
static PaintBuffer* m_inst;
QPixmap m_buf;
bool m_overflow;
bool m_grabbed;
int m_timer;
int m_resetWidth;
int m_resetHeight;
};
PaintBuffer *PaintBuffer::m_inst = 0;
static void copyWidget(const QRect& r, QPainter *p, QWidget *widget, int tx, int ty)
{
if (r.isNull() || r.isEmpty() )
return;
QRegion blit(r);
QValueVector<QWidget*> cw;
QValueVector<QRect> cr;
if (widget->children()) {
// build region
QObjectListIterator it = *widget->children();
for (; it.current(); ++it) {
QWidget* const w = ::qt_cast<QWidget *>(it.current());
if ( w && !w->isTopLevel() && !w->isHidden()) {
QRect r2 = w->geometry();
blit -= r2;
r2 = r2.intersect( r );
r2.moveBy(-w->x(), -w->y());
cr.append(r2);
cw.append(w);
}
}
}
QMemArray<QRect> br = blit.rects();
const int cnt = br.size();
const bool external = p->device()->isExtDev();
QPixmap* const pm = PaintBuffer::grab( widget->size() );
if (!pm)
{
kdWarning(6040) << "Rendering widget [ " << widget->className() << " ] failed due to invalid size." << endl;
return;
}
// fill background
if ( external ) {
// even hackier!
QPainter pt( pm );
const QColor c = widget->colorGroup().base();
for (int i = 0; i < cnt; ++i)
pt.fillRect( br[i], c );
} else {
QRect dr;
for (int i = 0; i < cnt; ++i ) {
dr = br[i];
dr.moveBy( tx, ty );
dr = p->xForm( dr );
bitBlt(pm, br[i].topLeft(), p->device(), dr);
}
}
// send paint event
QPainter::redirect(widget, pm);
QPaintEvent e( r, false );
QApplication::sendEvent( widget, &e );
QPainter::redirect(widget, 0);
// transfer result
if ( external )
for ( int i = 0; i < cnt; ++i )
p->drawPixmap(QPoint(tx+br[i].x(), ty+br[i].y()), *pm, br[i]);
else
for ( int i = 0; i < cnt; ++i )
bitBlt(p->device(), p->xForm( QPoint(tx, ty) + br[i].topLeft() ), pm, br[i]);
// cleanup and recurse
PaintBuffer::release();
QValueVector<QWidget*>::iterator cwit = cw.begin();
QValueVector<QWidget*>::iterator cwitEnd = cw.end();
QValueVector<QRect>::const_iterator crit = cr.begin();
for (; cwit != cwitEnd; ++cwit, ++crit)
copyWidget(*crit, p, *cwit, tx+(*cwit)->x(), ty+(*cwit)->y());
}
void RenderWidget::paintWidget(PaintInfo& pI, QWidget *widget, int tx, int ty)
{
QPainter* const p = pI.p;
allowWidgetPaintEvents = true;
const bool dsbld = QSharedDoubleBuffer::isDisabled();
QSharedDoubleBuffer::setDisabled(true);
QRect rr = pI.r;
rr.moveBy(-tx, -ty);
const QRect r = widget->rect().intersect( rr );
copyWidget(r, p, widget, tx, ty);
QSharedDoubleBuffer::setDisabled(dsbld);
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 ( ::qt_cast<KHTMLView *>( m_widget ) )
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;
//kdDebug() << "RenderWidget::eventFilter type=" << e->type() << endl;
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 ( QFocusEvent::reason() != QFocusEvent::Popup )
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!
//kdDebug(6000) << "RenderWidget::eventFilter captures FocusIn" << endl;
element()->getDocument()->setFocusNode(element());
// if ( isEditable() ) {
// KHTMLPartBrowserExtension *ext = static_cast<KHTMLPartBrowserExtension *>( element()->view->part()->browserExtension() );
// if ( ext ) ext->editableWidgetFocused( m_widget );
// }
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;
case QEvent::Wheel:
if (widget()->parentWidget() == view()->viewport()) {
// don't allow the widget to react to wheel event unless its
// currently focused. this avoids accidentally changing a select box
// or something while wheeling a webpage.
if (qApp->focusWidget() != widget() &&
widget()->focusPolicy() <= QWidget::StrongFocus) {
static_cast<QWheelEvent*>(e)->ignore();
QApplication::sendEvent(view(), e);
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) {
switch(e->type()) {
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:
keyReleaseEvent(static_cast<QKeyEvent *>(e));
break;
default:
break;
}
}
void RenderWidget::ScrollViewEventPropagator::sendEvent(QEvent *e) {
switch(e->type()) {
case QEvent::MouseButtonPress:
viewportMousePressEvent(static_cast<QMouseEvent *>(e));
break;
case QEvent::MouseButtonRelease:
viewportMouseReleaseEvent(static_cast<QMouseEvent *>(e));
break;
case QEvent::MouseButtonDblClick:
viewportMouseDoubleClickEvent(static_cast<QMouseEvent *>(e));
break;
case QEvent::MouseMove:
viewportMouseMoveEvent(static_cast<QMouseEvent *>(e));
break;
case QEvent::KeyPress:
keyPressEvent(static_cast<QKeyEvent *>(e));
break;
case QEvent::KeyRelease:
keyReleaseEvent(static_cast<QKeyEvent *>(e));
break;
default:
break;
}
}
bool RenderWidget::handleEvent(const DOM::EventImpl& ev)
{
bool ret = false;
switch(ev.id()) {
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();
int absx = 0;
int absy = 0;
absolutePosition(absx, absy);
QPoint p(me.clientX() - absx + m_view->contentsX(),
me.clientY() - absy + m_view->contentsY());
QMouseEvent::Type type;
int button = 0;
int state = 0;
if (qme) {
button = qme->button();
state = qme->state();
type = qme->type();
} else {
switch(me.id()) {
case EventImpl::MOUSEDOWN_EVENT:
type = QMouseEvent::MouseButtonPress;
break;
case EventImpl::MOUSEUP_EVENT:
type = QMouseEvent::MouseButtonRelease;
break;
case EventImpl::MOUSEMOVE_EVENT:
default:
type = QMouseEvent::MouseMove;
break;
}
switch (me.button()) {
case 0:
button = LeftButton;
break;
case 1:
button = MidButton;
break;
case 2:
button = RightButton;
break;
default:
break;
}
if (me.ctrlKey())
state |= ControlButton;
if (me.altKey())
state |= AltButton;
if (me.shiftKey())
state |= ShiftButton;
if (me.metaKey())
state |= MetaButton;
}
// kdDebug(6000) << "sending event to widget "
// << " pos=" << p << " type=" << type
// << " button=" << button << " state=" << state << endl;
QMouseEvent e(type, p, button, state);
QScrollView * sc = ::qt_cast<QScrollView*>(m_widget);
if (sc && !::qt_cast<QListBox*>(m_widget))
static_cast<ScrollViewEventPropagator *>(sc)->sendEvent(&e);
else
static_cast<EventPropagator *>(m_widget)->sendEvent(&e);
ret = e.isAccepted();
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->ascii(), ke->state(),
ke->text(), ke->isAutoRepeat(), ke->count() );
static_cast<EventPropagator *>(m_widget)->sendEvent(&releaseEv);
}
static_cast<EventPropagator *>(m_widget)->sendEvent(ke);
ret = ke->isAccepted();
break;
}
case EventImpl::MOUSEOUT_EVENT: {
QEvent moe( QEvent::Leave );
QApplication::sendEvent(m_widget, &moe);
break;
}
case EventImpl::MOUSEOVER_EVENT: {
QEvent moe( QEvent::Enter );
QApplication::sendEvent(m_widget, &moe);
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) {
khtml::SharedPtr<RenderArena> guard(m_arena); //Since delete on us gets called -first-,
//before the arena free
arenaDelete(m_arena.get());
}
}
FindSelectionResult RenderReplaced::checkSelectionPoint(int _x, int _y, int _tx, int _ty, DOM::NodeImpl*& node, int &offset, SelPointState &)
{
#if 0
kdDebug(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;
}
#ifdef ENABLE_DUMP
void RenderWidget::dump(QTextStream &stream, const QString &ind) const
{
RenderReplaced::dump(stream,ind);
if ( widget() )
stream << " color=" << widget()->foregroundColor().name()
<< " bg=" << widget()->backgroundColor().name();
else
stream << " null widget";
}
#endif
#include "render_replaced.moc"