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.
981 lines
28 KiB
981 lines
28 KiB
/*
|
|
Copyright (C) 2001-2003 KSVG Team
|
|
This file is part of the KDE project
|
|
|
|
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 <kdebug.h>
|
|
|
|
#include <tqtimer.h>
|
|
#include <tqstringlist.h>
|
|
#include <tqdatetime.h>
|
|
|
|
#define USE_VALGRIND 0
|
|
|
|
#if USE_VALGRIND
|
|
#include <valgrind/calltree.h>
|
|
#endif
|
|
|
|
#include "SVGLength.h"
|
|
|
|
#include "SVGRectImpl.h"
|
|
#include "SVGAngleImpl.h"
|
|
#include "SVGShapeImpl.h"
|
|
#include "SVGPointImpl.h"
|
|
#include "SVGNumberImpl.h"
|
|
#include "SVGMatrixImpl.h"
|
|
|
|
#include "SVGEventImpl.h"
|
|
#include "SVGAElementImpl.h"
|
|
#include "SVGDocumentImpl.h"
|
|
#include "SVGViewSpecImpl.h"
|
|
#include "SVGTransformImpl.h"
|
|
#include "SVGSVGElementImpl.h"
|
|
#include "SVGAnimatedRectImpl.h"
|
|
#include "SVGTransformListImpl.h"
|
|
#include "SVGAnimatedLengthImpl.h"
|
|
#include "SVGAnimatedStringImpl.h"
|
|
#include "SVGPreserveAspectRatioImpl.h"
|
|
#include "SVGAnimatedTransformListImpl.h"
|
|
#include "SVGAnimatedPreserveAspectRatioImpl.h"
|
|
|
|
#include "CanvasItem.h"
|
|
#include "KSVGCanvas.h"
|
|
|
|
using namespace KSVG;
|
|
|
|
#include "SVGSVGElementImpl.lut.h"
|
|
#include "ksvg_scriptinterpreter.h"
|
|
#include "ksvg_bridge.h"
|
|
#include "ksvg_ecma.h"
|
|
|
|
SVGSVGElementImpl::SVGSVGElementImpl(DOM::ElementImpl *impl) : SVGContainerImpl(impl), SVGTestsImpl(), SVGLangSpaceImpl(), SVGExternalResourcesRequiredImpl(), SVGStylableImpl(this), SVGLocatableImpl(), SVGFitToViewBoxImpl(), SVGZoomAndPanImpl()
|
|
{
|
|
KSVG_EMPTY_FLAGS
|
|
|
|
m_x = new SVGAnimatedLengthImpl(LENGTHMODE_WIDTH, this);
|
|
m_x->ref();
|
|
|
|
m_y = new SVGAnimatedLengthImpl(LENGTHMODE_HEIGHT, this);
|
|
m_y->ref();
|
|
|
|
m_width = new SVGAnimatedLengthImpl(LENGTHMODE_WIDTH, this);
|
|
m_width->ref();
|
|
|
|
m_height = new SVGAnimatedLengthImpl(LENGTHMODE_HEIGHT, this);
|
|
m_height->ref();
|
|
|
|
m_viewport = SVGSVGElementImpl::createSVGRect();
|
|
|
|
m_currentTranslate = SVGSVGElementImpl::createSVGPoint();
|
|
|
|
m_currentView = new SVGViewSpecImpl();
|
|
m_currentView->ref();
|
|
|
|
m_currentScale = 1.0;
|
|
|
|
m_useCurrentView = false;
|
|
|
|
m_clip[0] = 0;
|
|
m_clip[1] = 0;
|
|
m_clip[2] = 0;
|
|
m_clip[3] = 0;
|
|
|
|
m_rootParentScreenCTM = 0;
|
|
|
|
m_localMatrix = SVGSVGElementImpl::createSVGMatrix();
|
|
}
|
|
|
|
SVGSVGElementImpl::~SVGSVGElementImpl()
|
|
{
|
|
if(m_x)
|
|
m_x->deref();
|
|
if(m_y)
|
|
m_y->deref();
|
|
if(m_width)
|
|
m_width->deref();
|
|
if(m_height)
|
|
m_height->deref();
|
|
if(m_viewport)
|
|
m_viewport->deref();
|
|
if(m_currentTranslate)
|
|
m_currentTranslate->deref();
|
|
if(m_currentView)
|
|
m_currentView->deref();
|
|
if(m_rootParentScreenCTM)
|
|
m_rootParentScreenCTM->deref();
|
|
if(m_localMatrix)
|
|
m_localMatrix->deref();
|
|
}
|
|
|
|
bool SVGSVGElementImpl::isRootElement() const
|
|
{
|
|
return ownerDoc()->rootElement() == this;
|
|
}
|
|
|
|
void SVGSVGElementImpl::setAttributes()
|
|
{
|
|
SVGElementImpl::setAttributes();
|
|
|
|
// Spec: if not specified, effect is as if a value of "0" were specified
|
|
if(KSVG_TOKEN_NOT_PARSED(X))
|
|
KSVG_SET_ALT_ATTRIBUTE(X, "0")
|
|
|
|
// Spec: if not specified, effect is as if a value of "0" were specified
|
|
if(KSVG_TOKEN_NOT_PARSED(Y))
|
|
KSVG_SET_ALT_ATTRIBUTE(Y, "0")
|
|
|
|
// Spec: If the attribute is not specified, the effect is as if a value of "100%" were specified.
|
|
if(KSVG_TOKEN_NOT_PARSED(Width))
|
|
KSVG_SET_ALT_ATTRIBUTE(Width, "100%")
|
|
|
|
// Spec: If the attribute is not specified, the effect is as if a value of "100%" were specified.
|
|
if(KSVG_TOKEN_NOT_PARSED(Height))
|
|
KSVG_SET_ALT_ATTRIBUTE(Height, "100%")
|
|
|
|
// Spec: The contentScriptType should default to "text/ecmascript".
|
|
if(KSVG_TOKEN_NOT_PARSED(ContentScriptType))
|
|
KSVG_SET_ALT_ATTRIBUTE(ContentScriptType, "text/ecmascript")
|
|
|
|
// Spec: The contentStyleType should default to "text/css".
|
|
if(KSVG_TOKEN_NOT_PARSED(ContentStyleType))
|
|
KSVG_SET_ALT_ATTRIBUTE(ContentStyleType, "text/css")
|
|
|
|
if(m_useCurrentView)
|
|
{
|
|
parseViewBox(m_currentView->viewBoxString().string());
|
|
preserveAspectRatio()->baseVal()->parsePreserveAspectRatio(m_currentView->preserveAspectRatioString().string());
|
|
}
|
|
|
|
m_viewport->setX(x()->baseVal()->value());
|
|
m_viewport->setY(y()->baseVal()->value());
|
|
m_viewport->setWidth(width()->baseVal()->value());
|
|
m_viewport->setHeight(height()->baseVal()->value());
|
|
|
|
if(isRootElement() && ownerDoc()->parentImage() == 0)
|
|
{
|
|
if(ownerDoc()->canvas())
|
|
ownerDoc()->canvas()->setViewportDimension(int(ceil(width()->baseVal()->value() * currentScale())), int(ceil(height()->baseVal()->value() * currentScale())));
|
|
|
|
// Special case for outermost svg element:
|
|
// We need to register our id manually, because
|
|
// m_ownerSVGElement is 0 in SVGElementImpl::setAttributes (Niko)
|
|
if(!id().isNull())
|
|
addToIdMap(id().string(), this);
|
|
}
|
|
}
|
|
|
|
SVGAnimatedLengthImpl *SVGSVGElementImpl::x()
|
|
{
|
|
return m_x;
|
|
}
|
|
|
|
SVGAnimatedLengthImpl *SVGSVGElementImpl::y()
|
|
{
|
|
return m_y;
|
|
}
|
|
|
|
SVGAnimatedLengthImpl *SVGSVGElementImpl::width()
|
|
{
|
|
return m_width;
|
|
}
|
|
|
|
SVGAnimatedLengthImpl *SVGSVGElementImpl::height()
|
|
{
|
|
return m_height;
|
|
}
|
|
|
|
void SVGSVGElementImpl::setContentScriptType(const DOM::DOMString &contentScriptType)
|
|
{
|
|
setAttribute("contentScriptType", contentScriptType);
|
|
}
|
|
|
|
DOM::DOMString SVGSVGElementImpl::contentScriptType() const
|
|
{
|
|
return getAttribute("contentScriptType");
|
|
}
|
|
|
|
void SVGSVGElementImpl::setContentStyleType(const DOM::DOMString &contentStyleType)
|
|
{
|
|
setAttribute("contentStyleType", contentStyleType);
|
|
}
|
|
|
|
DOM::DOMString SVGSVGElementImpl::contentStyleType() const
|
|
{
|
|
return getAttribute("contentStyleType");
|
|
}
|
|
|
|
SVGRectImpl *SVGSVGElementImpl::viewport()
|
|
{
|
|
return m_viewport;
|
|
}
|
|
|
|
SVGRectImpl *SVGSVGElementImpl::getBBox()
|
|
{
|
|
SVGRectImpl *ret = new SVGRectImpl(getCTM()->qmatrix().invert().map(m_viewport->qrect()));
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
float SVGSVGElementImpl::pixelUnitToMillimeterX() const
|
|
{
|
|
return ownerDoc()->screenPixelsPerMillimeterX();
|
|
}
|
|
|
|
float SVGSVGElementImpl::pixelUnitToMillimeterY() const
|
|
{
|
|
return ownerDoc()->screenPixelsPerMillimeterY();
|
|
}
|
|
|
|
float SVGSVGElementImpl::screenPixelToMillimeterX() const
|
|
{
|
|
return pixelUnitToMillimeterX();
|
|
}
|
|
|
|
float SVGSVGElementImpl::screenPixelToMillimeterY() const
|
|
{
|
|
return pixelUnitToMillimeterY();
|
|
}
|
|
|
|
void SVGSVGElementImpl::setUseCurrentView(bool useCurrentView)
|
|
{
|
|
m_useCurrentView = useCurrentView;
|
|
}
|
|
|
|
bool SVGSVGElementImpl::useCurrentView() const
|
|
{
|
|
return m_useCurrentView;
|
|
}
|
|
|
|
SVGViewSpecImpl *SVGSVGElementImpl::currentView() const
|
|
{
|
|
return m_currentView;
|
|
}
|
|
|
|
void SVGSVGElementImpl::setCurrentScale(float currentScale)
|
|
{
|
|
if( m_currentScale != currentScale )
|
|
{
|
|
m_currentScale = currentScale;
|
|
invalidateCachedMatrices();
|
|
|
|
if(hasEventListener(SVGEvent::ZOOM_EVENT, true))
|
|
dispatchEvent(SVGEvent::ZOOM_EVENT, false, false);
|
|
}
|
|
}
|
|
|
|
float SVGSVGElementImpl::currentScale() const
|
|
{
|
|
return m_currentScale;
|
|
}
|
|
|
|
void SVGSVGElementImpl::setCurrentTranslate(const TQPoint &p)
|
|
{
|
|
if(m_currentTranslate->x() != p.x() || m_currentTranslate->y() != p.y())
|
|
{
|
|
m_currentTranslate->setX(p.x());
|
|
m_currentTranslate->setY(p.y());
|
|
invalidateCachedMatrices();
|
|
if(hasEventListener(SVGEvent::SCROLL_EVENT, true))
|
|
dispatchEvent(SVGEvent::SCROLL_EVENT, false, false);
|
|
}
|
|
}
|
|
|
|
SVGPointImpl *SVGSVGElementImpl::currentTranslate()
|
|
{
|
|
return m_currentTranslate;
|
|
}
|
|
|
|
unsigned long SVGSVGElementImpl::suspendRedraw(unsigned long)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void SVGSVGElementImpl::unsuspendRedraw(unsigned long)
|
|
{
|
|
}
|
|
|
|
void SVGSVGElementImpl::unsuspendRedrawAll()
|
|
{
|
|
}
|
|
|
|
void SVGSVGElementImpl::forceRedraw()
|
|
{
|
|
#if USE_VALGRIND
|
|
CALLTREE_ZERO_STATS();
|
|
#endif
|
|
|
|
TQTime timer;
|
|
timer.start();
|
|
|
|
if(ownerDoc() && ownerDoc()->canvas())
|
|
ownerDoc()->canvas()->update();
|
|
|
|
kdDebug(26000) << "forceRedraw in " << timer.elapsed()/1000.0 << " seconds" << endl;
|
|
|
|
#if USE_VALGRIND
|
|
CALLTREE_DUMP_STATS();
|
|
#endif
|
|
}
|
|
|
|
void SVGSVGElementImpl::pauseAnimations()
|
|
{
|
|
if(!ownerDoc()->timeScheduler()->animationsPaused())
|
|
ownerDoc()->timeScheduler()->toggleAnimations();
|
|
}
|
|
|
|
void SVGSVGElementImpl::unpauseAnimations()
|
|
{
|
|
if(ownerDoc()->timeScheduler()->animationsPaused())
|
|
ownerDoc()->timeScheduler()->toggleAnimations();
|
|
}
|
|
|
|
bool SVGSVGElementImpl::animationsPaused()
|
|
{
|
|
return ownerDoc()->timeScheduler()->animationsPaused();
|
|
}
|
|
|
|
float SVGSVGElementImpl::getCurrentTime() const
|
|
{
|
|
return ownerDoc()->timeScheduler()->elapsed();
|
|
}
|
|
|
|
void SVGSVGElementImpl::setCurrentTime(float)
|
|
{
|
|
}
|
|
|
|
DOM::NodeList SVGSVGElementImpl::getIntersectionList(SVGRectImpl *, SVGElementImpl *)
|
|
{
|
|
// TODO : implement me
|
|
return DOM::NodeList();
|
|
}
|
|
|
|
DOM::NodeList SVGSVGElementImpl::getEnclosureList(SVGRectImpl *rect, SVGElementImpl */*referenceElement*/)
|
|
{
|
|
DOM::NodeList list;
|
|
|
|
DOM::Node node = firstChild();
|
|
for(; !node.isNull(); node = node.nextSibling())
|
|
{
|
|
SVGElementImpl *element = ownerDoc()->getElementFromHandle(node.handle());
|
|
SVGShapeImpl *shape = dynamic_cast<SVGShapeImpl *>(element);
|
|
if(shape)
|
|
{
|
|
if(shape->isContainer())
|
|
// TODO : pass it on to container::getEnclosureList() which should return a NodeList
|
|
kdDebug() << "!shape" << endl;
|
|
else
|
|
{
|
|
// TODO : add the shape to list if the test succeeds
|
|
SVGRectImpl *current = shape->getBBox();
|
|
if(rect->qrect().contains(current->qrect(), true))
|
|
kdDebug() << "shape : " << element->nodeName().string() << " is fully enclosed" << endl;
|
|
|
|
current->deref();
|
|
}
|
|
}
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
bool SVGSVGElementImpl::checkIntersection(SVGElementImpl *element, SVGRectImpl *rect)
|
|
{
|
|
SVGShapeImpl *shape = dynamic_cast<SVGShapeImpl *>(element);
|
|
if(!shape)
|
|
return false;
|
|
|
|
SVGRectImpl *current = shape->getBBox();
|
|
bool result = rect->qrect().intersects(current->qrect());
|
|
current->deref();
|
|
return result;
|
|
}
|
|
|
|
bool SVGSVGElementImpl::checkEnclosure(SVGElementImpl *element, SVGRectImpl *rect)
|
|
{
|
|
SVGShapeImpl *shape = dynamic_cast<SVGShapeImpl *>(element);
|
|
if(!shape)
|
|
return false;
|
|
|
|
SVGRectImpl *current = shape->getBBox();
|
|
bool result = rect->qrect().contains(current->qrect());
|
|
current->deref();
|
|
return result;
|
|
}
|
|
|
|
void SVGSVGElementImpl::deSelectAll()
|
|
{
|
|
}
|
|
|
|
SVGNumberImpl *SVGSVGElementImpl::createSVGNumber()
|
|
{
|
|
// Spec: Creates an SVGNumber object outside of any document
|
|
// trees. The object is initialized to a value of zero.
|
|
SVGNumberImpl *ret = new SVGNumberImpl();
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGLengthImpl *SVGSVGElementImpl::createSVGLength()
|
|
{
|
|
// Spec: Creates an SVGLength object outside of any document
|
|
// trees. The object is initialized to the value of 0 user units.
|
|
SVGLengthImpl *ret = new SVGLengthImpl();
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGAngleImpl *SVGSVGElementImpl::createSVGAngle()
|
|
{
|
|
// Spec: Creates an SVGAngle object outside of any document
|
|
// trees. The object is initialized to the value 0 degrees (unitless).
|
|
SVGAngleImpl *ret = new SVGAngleImpl();
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGPointImpl *SVGSVGElementImpl::createSVGPoint()
|
|
{
|
|
// Spec: Creates an SVGPoint object outside of any document
|
|
// trees. The object is initialized to the point (0,0) in the user coordinate system.
|
|
SVGPointImpl *ret = new SVGPointImpl();
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGMatrixImpl *SVGSVGElementImpl::createSVGMatrix()
|
|
{
|
|
// Spec: Creates an SVGMatrix object outside of any document
|
|
// trees. The object is initialized to the identity matrix.
|
|
SVGMatrixImpl *ret = new SVGMatrixImpl(TQWMatrix(1.0F, 0.0F, 0.0F, 1.0F, 0.0F, 0.0F));
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGRectImpl *SVGSVGElementImpl::createSVGRect()
|
|
{
|
|
// Spec: Creates an SVGRect object outside of any document
|
|
// trees. The object is initialized such that all values are set to 0 user units.
|
|
SVGRectImpl *ret = new SVGRectImpl();
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGTransformImpl *SVGSVGElementImpl::createSVGTransform()
|
|
{
|
|
// Spec: Creates an SVGTransform object outside of any document
|
|
// trees. The object is initialized to an identity matrix transform (SVG_TRANSFORM_MATRIX).
|
|
SVGTransformImpl *transform = createSVGTransformFromMatrix(createSVGMatrix());
|
|
|
|
// createSVGMatrix already ref's the matrix, the SVGTransformImpl->setMatrix
|
|
// call also does this, prevent non deleting of the object by deref'ing (Niko)
|
|
transform->matrix()->deref();
|
|
|
|
return transform;
|
|
}
|
|
|
|
SVGTransformImpl *SVGSVGElementImpl::createSVGTransformFromMatrix(SVGMatrixImpl *mat)
|
|
{
|
|
// Spec: Creates an SVGTransform object outside of any document
|
|
// trees. The object is initialized to the given matrix transform (i.e., SVG_TRANSFORM_MATRIX).
|
|
SVGTransformImpl *ret = new SVGTransformImpl();
|
|
ret->setMatrix(mat);
|
|
ret->ref();
|
|
return ret;
|
|
}
|
|
|
|
SVGElementImpl *SVGSVGElementImpl::getElementById(const DOM::DOMString &elementId)
|
|
{
|
|
return m_map[elementId.string()];
|
|
}
|
|
|
|
void SVGSVGElementImpl::addToIdMap(const TQString &id, SVGElementImpl *obj)
|
|
{
|
|
m_map.insert(id, obj);
|
|
}
|
|
|
|
SVGMatrixImpl *SVGSVGElementImpl::getCTM()
|
|
{
|
|
return viewBoxToViewTransform(width()->baseVal()->value(), height()->baseVal()->value());
|
|
}
|
|
|
|
const SVGMatrixImpl *SVGSVGElementImpl::localMatrix()
|
|
{
|
|
// TODO: only update the matrix when needed and just return m_localMatrix
|
|
|
|
m_localMatrix->reset();
|
|
|
|
if(ownerSVGElement() == 0)
|
|
{
|
|
if(m_rootParentScreenCTM != 0)
|
|
m_localMatrix->copy(m_rootParentScreenCTM);
|
|
|
|
// We're the outermost svg element.
|
|
// Put the zoom scale and translate into the matrix.
|
|
m_localMatrix->translate(currentTranslate()->x(), currentTranslate()->y());
|
|
m_localMatrix->scale(currentScale());
|
|
}
|
|
|
|
// Apply viewport translation.
|
|
m_localMatrix->translate(x()->baseVal()->value(), y()->baseVal()->value());
|
|
|
|
// And viewbox.
|
|
SVGMatrixImpl *viewboxMatrix = viewBoxToViewTransform(width()->baseVal()->value(), height()->baseVal()->value());
|
|
|
|
m_localMatrix->multiply(viewboxMatrix);
|
|
viewboxMatrix->deref();
|
|
|
|
return m_localMatrix;
|
|
}
|
|
|
|
void SVGSVGElementImpl::setClip(const TQString &clip)
|
|
{
|
|
// TODO : this routine should probably be shared between all classes that establish new viewports (Rob)
|
|
if(!clip.startsWith("rect(") || !clip.endsWith(")"))
|
|
return;
|
|
|
|
TQString work = clip.mid(5, clip.length() - 6);
|
|
TQStringList substrings = TQStringList::split(',', clip);
|
|
TQStringList::ConstIterator it = substrings.begin();
|
|
|
|
if(m_clip[0])
|
|
m_clip[0]->deref();
|
|
m_clip[0] = SVGSVGElementImpl::createSVGLength();
|
|
|
|
if(*it != "auto")
|
|
m_clip[0]->setValueAsString(*it);
|
|
++it;
|
|
|
|
if(m_clip[1])
|
|
m_clip[1]->deref();
|
|
m_clip[1] = SVGSVGElementImpl::createSVGLength();
|
|
|
|
if(*it != "auto")
|
|
m_clip[1]->setValueAsString(*it);
|
|
++it;
|
|
|
|
if(m_clip[2])
|
|
m_clip[2]->deref();
|
|
m_clip[2] = SVGSVGElementImpl::createSVGLength();
|
|
|
|
if(*it != "auto")
|
|
m_clip[2]->setValueAsString(*it);
|
|
++it;
|
|
|
|
if(m_clip[3])
|
|
m_clip[3]->deref();
|
|
m_clip[3] = SVGSVGElementImpl::createSVGLength();
|
|
|
|
if(*it != "auto")
|
|
m_clip[3]->setValueAsString(*it);
|
|
}
|
|
|
|
TQRect SVGSVGElementImpl::clip()
|
|
{
|
|
// Get viewport in user coordinates.
|
|
TQRect v(0, 0, m_viewport->qrect().width(), m_viewport->qrect().height());
|
|
|
|
SVGMatrixImpl *ctm = getCTM();
|
|
TQRect r = ctm->qmatrix().invert().mapRect(v);
|
|
ctm->deref();
|
|
|
|
if(m_clip[0])
|
|
r.setTop(static_cast<int>(r.top() + m_clip[0]->value()));
|
|
if(m_clip[1])
|
|
r.setRight(static_cast<int>(r.right() - m_clip[1]->value()));
|
|
if(m_clip[2])
|
|
r.setBottom(static_cast<int>(r.bottom() - m_clip[2]->value()));
|
|
if(m_clip[3])
|
|
r.setLeft(static_cast<int>(r.left() + m_clip[3]->value()));
|
|
|
|
return r;
|
|
}
|
|
|
|
void SVGSVGElementImpl::setRootParentScreenCTM(SVGMatrixImpl *screenCTM)
|
|
{
|
|
if(m_rootParentScreenCTM != 0)
|
|
m_rootParentScreenCTM->deref();
|
|
|
|
m_rootParentScreenCTM = screenCTM;
|
|
screenCTM->ref();
|
|
}
|
|
|
|
bool SVGSVGElementImpl::prepareMouseEvent(const TQPoint &p, const TQPoint &a, SVGMouseEventImpl *mev)
|
|
{
|
|
// mop: central bool var which turns to true once the current "mouseover" element has been found
|
|
bool ret = false, dorerender = false;
|
|
SVGElementImpl *elem = 0;
|
|
|
|
SVGMatrixImpl *ctm = getCTM();
|
|
TQPoint userA = ctm->qmatrix().invert().map(a);
|
|
ctm->deref();
|
|
|
|
// Just check the lastTarget once (mop)
|
|
if(ownerDoc()->lastTarget())
|
|
{
|
|
elem = ownerDoc()->lastTarget();
|
|
ret = elem->prepareMouseEvent(p, userA, mev);
|
|
|
|
// mop: only proceed if element is not the current element anymore. rest is done in the lower part
|
|
if(!ret)
|
|
{
|
|
if(elem->hasEventListener(SVGEvent::MOUSEOUT_EVENT, false))
|
|
{
|
|
dorerender = true;
|
|
elem->setMouseOver(false);
|
|
elem->dispatchMouseEvent(SVGEvent::MOUSEOUT_EVENT, true, true, 0, mev->screenX(), mev->screenY(), mev->clientX(), mev->clientY(), mev->ctrlKey(), mev->altKey(), mev->shiftKey(), mev->metaKey(), mev->button(), elem);
|
|
}
|
|
|
|
if(elem->hasEventListener(SVGEvent::DOMFOCUSOUT_EVENT, false) && elem->focus())
|
|
{
|
|
dorerender = true;
|
|
elem->setFocus(false);
|
|
elem->dispatchEvent(SVGEvent::DOMFOCUSOUT_EVENT, true, true);
|
|
}
|
|
|
|
// mop: unset last target once we left it
|
|
ownerDoc()->setLastTarget(0);
|
|
}
|
|
}
|
|
|
|
// mop: DAMN...logic doesn't work :(
|
|
// we cant use the results of the above check because something like this
|
|
// _________
|
|
// | ____ |
|
|
// ||____| |
|
|
// |_________|
|
|
//
|
|
// wouldn't work because even if the mousepointer would be in the inner rect it would still be inside the bbox
|
|
// of the outer (assuming that the outer is the lastTarget). :(
|
|
ret = false;
|
|
|
|
// mop: just check for a node until the element where the mouse is over is found
|
|
// mop: "ret" can be set from the above stuff too...
|
|
// find the element here and do the event stuff in the lower part..much cleaner
|
|
CanvasItemList hits = ownerDoc()->canvas()->collisions(p, true);
|
|
for(CanvasItemList::Iterator it = hits.begin(); it != hits.end(); ++it)
|
|
{
|
|
elem = (*it)->element();
|
|
if(elem)
|
|
{
|
|
// Check if mouse is over a certain shape...
|
|
// mop: once an element has been found check eventlisteners and leave immediately
|
|
ret = elem->prepareMouseEvent(p, userA, mev);
|
|
if(ret) break;
|
|
}
|
|
}
|
|
|
|
// mop: has an element been found?
|
|
if(ret)
|
|
{
|
|
int events = mev->target()->getEventListeners(false);
|
|
|
|
// Dispatch mousemove, mousedown, mouseup and mouseclick events
|
|
bool cancel = (mev->id() != SVGEvent::MOUSEMOVE_EVENT);
|
|
|
|
if(events & 1 << mev->id())
|
|
{
|
|
mev->target()->dispatchMouseEvent(mev->id(), true, cancel, 0, mev->screenX(), mev->screenY(), mev->clientX(), mev->clientY(), mev->ctrlKey(), mev->altKey(), mev->shiftKey(), mev->metaKey(), mev->button(), elem);
|
|
dorerender = true; // mop: if it has the event then rerender
|
|
}
|
|
|
|
// If a mouse "moves" over a shape, it's also "over" the shape
|
|
if(mev->id() == SVGEvent::MOUSEMOVE_EVENT)
|
|
{
|
|
mev->target()->setMouseOver(true);
|
|
if(events & 1 << SVGEvent::MOUSEOVER_EVENT)
|
|
{
|
|
mev->target()->dispatchMouseEvent(SVGEvent::MOUSEOVER_EVENT, true, cancel, 0, mev->screenX(), mev->screenY(), mev->clientX(), mev->clientY(), mev->ctrlKey(), mev->altKey(), mev->shiftKey(), mev->metaKey(), mev->button(), elem);
|
|
dorerender = true;
|
|
}
|
|
|
|
}
|
|
|
|
// Also send an domactivate + focusin event on mouseup
|
|
bool dolinktest = true;
|
|
if(mev->id() == SVGEvent::MOUSEUP_EVENT)
|
|
{
|
|
mev->target()->setFocus(true);
|
|
|
|
if(events & 1 << SVGEvent::CLICK_EVENT)
|
|
{
|
|
dolinktest = mev->target()->dispatchEvent(SVGEvent::CLICK_EVENT, true, true);
|
|
dorerender = true;
|
|
}
|
|
|
|
if(events & 1 << SVGEvent::DOMACTIVATE_EVENT)
|
|
{
|
|
mev->target()->dispatchEvent(SVGEvent::DOMACTIVATE_EVENT, true, true);
|
|
dorerender = true;
|
|
}
|
|
|
|
if(events & 1 << SVGEvent::DOMFOCUSIN_EVENT)
|
|
{
|
|
mev->target()->dispatchEvent(SVGEvent::DOMFOCUSIN_EVENT, true, true);
|
|
dorerender = true;
|
|
}
|
|
}
|
|
|
|
// Hyperlink support
|
|
SVGAElementImpl* link=0;
|
|
if(dolinktest && !mev->defaultPrevented())
|
|
{
|
|
link = SVGAElementImpl::getLink(elem);
|
|
if(link)
|
|
{
|
|
mev->setURL(link->href()->baseVal());
|
|
emit ownerDoc()->gotURL(link->target()->baseVal().string());
|
|
}
|
|
}
|
|
|
|
// The mouse is over a shape, so we have a target..we need to register that for a mouseout
|
|
ownerDoc()->setLastTarget(mev->target());
|
|
}
|
|
|
|
// mop: all events may trigger changed style, add elements etc. this is definately needed :(
|
|
if(dorerender)
|
|
ownerDoc()->rerender();
|
|
|
|
return dorerender; // mop: some kind of event has been found and executed
|
|
}
|
|
|
|
// Ecma stuff
|
|
|
|
/*
|
|
@namespace KSVG
|
|
@begin SVGSVGElementImpl::s_hashTable 23
|
|
x SVGSVGElementImpl::X DontDelete|ReadOnly
|
|
y SVGSVGElementImpl::Y DontDelete|ReadOnly
|
|
width SVGSVGElementImpl::Width DontDelete|ReadOnly
|
|
height SVGSVGElementImpl::Height DontDelete|ReadOnly
|
|
viewport SVGSVGElementImpl::Viewport DontDelete|ReadOnly
|
|
contentScriptType SVGSVGElementImpl::ContentScriptType DontDelete
|
|
contentStyleType SVGSVGElementImpl::ContentStyleType DontDelete
|
|
pixelUnitToMillimeterX SVGSVGElementImpl::PixelUnitToMillimeterX DontDelete|ReadOnly
|
|
pixelUnitToMillimeterY SVGSVGElementImpl::PixelUnitToMillimeterY DontDelete|ReadOnly
|
|
screenPixelToMillimeterX SVGSVGElementImpl::ScreenPixelToMillimeterX DontDelete|ReadOnly
|
|
screenPixelToMillimeterY SVGSVGElementImpl::ScreenPixelToMillimeterY DontDelete|ReadOnly
|
|
useCurrentView SVGSVGElementImpl::UseCurrentView DontDelete
|
|
currentScale SVGSVGElementImpl::CurrentScale DontDelete
|
|
currentTranslate SVGSVGElementImpl::CurrentTranslate DontDelete|ReadOnly
|
|
onunload SVGSVGElementImpl::OnUnload DontDelete
|
|
onerror SVGSVGElementImpl::OnError DontDelete
|
|
onresize SVGSVGElementImpl::OnResize DontDelete
|
|
onzoom SVGSVGElementImpl::OnZoom DontDelete
|
|
onscroll SVGSVGElementImpl::OnScroll DontDelete
|
|
@end
|
|
@namespace KSVG
|
|
@begin SVGSVGElementImplProto::s_hashTable 29
|
|
createSVGNumber SVGSVGElementImpl::CreateSVGNumber DontDelete|Function 0
|
|
createSVGLength SVGSVGElementImpl::CreateSVGLength DontDelete|Function 0
|
|
createSVGAngle SVGSVGElementImpl::CreateSVGAngle DontDelete|Function 0
|
|
createSVGPoint SVGSVGElementImpl::CreateSVGPoint DontDelete|Function 0
|
|
createSVGMatrix SVGSVGElementImpl::CreateSVGMatrix DontDelete|Function 0
|
|
createSVGRect SVGSVGElementImpl::CreateSVGRect DontDelete|Function 0
|
|
createSVGTransform SVGSVGElementImpl::CreateSVGTransform DontDelete|Function 0
|
|
createSVGTransformFromMatrix SVGSVGElementImpl::CreateSVGTransformFromMatrix DontDelete|Function 1
|
|
suspendRedraw SVGSVGElementImpl::SuspendRedraw DontDelete|Function 1
|
|
unsuspendRedraw SVGSVGElementImpl::UnsuspendRedraw DontDelete|Function 1
|
|
unsuspendRedrawAll SVGSVGElementImpl::UnsuspendRedrawAll DontDelete|Function 0
|
|
forceRedraw SVGSVGElementImpl::ForceRedraw DontDelete|Function 0
|
|
pauseAnimations SVGSVGElementImpl::PauseAnimations DontDelete|Function 0
|
|
unpauseAnimations SVGSVGElementImpl::UnpauseAnimations DontDelete|Function 0
|
|
animationsPaused SVGSVGElementImpl::AnimationsPaused DontDelete|Function 0
|
|
getCurrentTime SVGSVGElementImpl::GetCurrentTime DontDelete|Function 0
|
|
setCurrentTime SVGSVGElementImpl::SetCurrentTime DontDelete|Function 1
|
|
getIntersectionList SVGSVGElementImpl::GetIntersectionList DontDelete|Function 2
|
|
getEnclosureList SVGSVGElementImpl::GetEnclosureList DontDelete|Function 2
|
|
checkIntersection SVGSVGElementImpl::CheckIntersection DontDelete|Function 2
|
|
checkEnclosure SVGSVGElementImpl::CheckEnclosure DontDelete|Function 2
|
|
deselectAll SVGSVGElementImpl::DeselectAll DontDelete|Function 0
|
|
getElementById SVGSVGElementImpl::GetElementById DontDelete|Function 1
|
|
@end
|
|
*/
|
|
|
|
KSVG_IMPLEMENT_PROTOTYPE("SVGSVGElement", SVGSVGElementImplProto, SVGSVGElementImplProtoFunc)
|
|
|
|
Value SVGSVGElementImpl::getValueProperty(ExecState *exec, int token) const
|
|
{
|
|
KSVG_CHECK_ATTRIBUTE
|
|
|
|
switch(token)
|
|
{
|
|
case X:
|
|
if(!attributeMode)
|
|
return m_x->cache(exec);
|
|
else
|
|
return Number(m_x->baseVal()->value());
|
|
case Y:
|
|
if(!attributeMode)
|
|
return m_y->cache(exec);
|
|
else
|
|
return Number(m_y->baseVal()->value());
|
|
case Width:
|
|
if(!attributeMode)
|
|
return m_width->cache(exec);
|
|
else
|
|
return Number(m_width->baseVal()->value());
|
|
case Height:
|
|
if(!attributeMode)
|
|
return m_height->cache(exec);
|
|
else
|
|
return Number(m_height->baseVal()->value());
|
|
case Viewport:
|
|
return m_viewport->cache(exec);
|
|
case ContentScriptType:
|
|
return String(contentScriptType().string());
|
|
case ContentStyleType:
|
|
return String(contentStyleType().string());
|
|
case PixelUnitToMillimeterX:
|
|
return Number(pixelUnitToMillimeterX());
|
|
case PixelUnitToMillimeterY:
|
|
return Number(pixelUnitToMillimeterY());
|
|
case ScreenPixelToMillimeterX:
|
|
return Number(screenPixelToMillimeterX());
|
|
case ScreenPixelToMillimeterY:
|
|
return Number(screenPixelToMillimeterY());
|
|
case UseCurrentView:
|
|
return Boolean(useCurrentView());
|
|
case CurrentScale:
|
|
return Number(currentScale());
|
|
case CurrentTranslate:
|
|
return m_currentTranslate->cache(exec);
|
|
default:
|
|
kdWarning() << "Unhandled token in " << k_funcinfo << " : " << token << endl;
|
|
return Undefined();
|
|
}
|
|
}
|
|
|
|
void SVGSVGElementImpl::putValueProperty(ExecState *exec, int token, const Value &value, int)
|
|
{
|
|
switch(token)
|
|
{
|
|
case ContentScriptType:
|
|
setContentScriptType(value.toString(exec).string());
|
|
break;
|
|
case ContentStyleType:
|
|
setContentStyleType(value.toString(exec).string());
|
|
break;
|
|
case CurrentScale:
|
|
m_currentScale = value.toNumber(exec);
|
|
break;
|
|
case X:
|
|
x()->baseVal()->setValueAsString(value.toString(exec).qstring());
|
|
break;
|
|
case Y:
|
|
y()->baseVal()->setValueAsString(value.toString(exec).qstring());
|
|
break;
|
|
case Width:
|
|
width()->baseVal()->setValueAsString(value.toString(exec).qstring());
|
|
break;
|
|
case Height:
|
|
height()->baseVal()->setValueAsString(value.toString(exec).qstring());
|
|
break;
|
|
case OnUnload:
|
|
// Spec: only applicable to outermost 'svg' elements
|
|
if(isRootElement())
|
|
setEventListener(SVGEvent::UNLOAD_EVENT, ownerDoc()->createEventListener(value.toString(exec).qstring()));
|
|
break;
|
|
case OnError:
|
|
setEventListener(SVGEvent::ERROR_EVENT, ownerDoc()->createEventListener(value.toString(exec).qstring()));
|
|
break;
|
|
case OnResize:
|
|
// Spec: only applicable to outermost 'svg' elements
|
|
if(isRootElement())
|
|
setEventListener(SVGEvent::RESIZE_EVENT, ownerDoc()->createEventListener(value.toString(exec).qstring()));
|
|
break;
|
|
case OnZoom:
|
|
// Spec: only applicable to outermost 'svg' elements
|
|
if(isRootElement())
|
|
setEventListener(SVGEvent::ZOOM_EVENT, ownerDoc()->createEventListener(value.toString(exec).qstring()));
|
|
break;
|
|
case OnScroll:
|
|
// Spec: only applicable to outermost 'svg' elements
|
|
if(isRootElement())
|
|
setEventListener(SVGEvent::SCROLL_EVENT, ownerDoc()->createEventListener(value.toString(exec).qstring()));
|
|
break;
|
|
default:
|
|
kdWarning() << k_funcinfo << "unhandled token " << token << endl;
|
|
}
|
|
}
|
|
|
|
Value SVGSVGElementImplProtoFunc::call(ExecState *exec, Object &thisObj, const List &args)
|
|
{
|
|
KSVG_CHECK_THIS(SVGSVGElementImpl)
|
|
|
|
switch(id)
|
|
{
|
|
case SVGSVGElementImpl::CreateSVGNumber:
|
|
return obj->createSVGNumber()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGLength:
|
|
return obj->createSVGLength()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGAngle:
|
|
return obj->createSVGAngle()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGPoint:
|
|
return obj->createSVGPoint()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGMatrix:
|
|
return obj->createSVGMatrix()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGRect:
|
|
return obj->createSVGRect()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGTransform:
|
|
return obj->createSVGTransform()->cache(exec);
|
|
case SVGSVGElementImpl::CreateSVGTransformFromMatrix:
|
|
return obj->createSVGTransformFromMatrix(static_cast<KSVGBridge<SVGMatrixImpl> *>(args[0].imp())->impl())->cache(exec);
|
|
case SVGSVGElementImpl::GetElementById:
|
|
{
|
|
// Keep in sync with SVGDocumentImpl's version.
|
|
Value ret;
|
|
|
|
SVGElementImpl *element = obj->getElementById(args[0].toString(exec).string());
|
|
|
|
if(element)
|
|
ret = getDOMNode(exec, *element);
|
|
else
|
|
{
|
|
element = obj->ownerDoc()->recursiveSearch(*(obj->ownerDoc()), args[0].toString(exec).string());
|
|
if(!element)
|
|
return Null();
|
|
|
|
ret = getDOMNode(exec, *element);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
case SVGSVGElementImpl::GetCurrentTime:
|
|
return Number(obj->getCurrentTime());
|
|
case SVGSVGElementImpl::SetCurrentTime:
|
|
obj->setCurrentTime(args[0].toNumber(exec));
|
|
return Undefined();
|
|
case SVGSVGElementImpl::DeselectAll:
|
|
obj->deSelectAll();
|
|
return Undefined();
|
|
case SVGSVGElementImpl::PauseAnimations:
|
|
obj->pauseAnimations();
|
|
return Undefined();
|
|
case SVGSVGElementImpl::UnpauseAnimations:
|
|
obj->unpauseAnimations();
|
|
return Undefined();
|
|
case SVGSVGElementImpl::AnimationsPaused:
|
|
return Boolean(obj->animationsPaused());
|
|
default:
|
|
kdWarning() << "Unhandled function id in " << k_funcinfo << " : " << id << endl;
|
|
break;
|
|
}
|
|
|
|
return Undefined();
|
|
}
|