/* Copyright (C) Research In Motion Limited 2010. All rights reserved. 2004, 2005, 2007 Nikolas Zimmermann <zimmermann@kde.org> 2004, 2005, 2008 Rob Buis <buis@kde.org> 2005, 2007 Eric Seidel <eric@webkit.org> 2009 Google, Inc. 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 aint with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #if ENABLE(SVG) #include "SVGMarkerLayoutInfo.h" #include "RenderSVGViewportContainer.h" #include "SVGResourceMarker.h" namespace WebCore { SVGMarkerLayoutInfo::SVGMarkerLayoutInfo() : m_midMarker(0) , m_elementIndex(0) , m_strokeWidth(0) { } SVGMarkerLayoutInfo::~SVGMarkerLayoutInfo() { } static inline void processStartAndMidMarkers(void* infoPtr, const PathElement* element) { SVGMarkerLayoutInfo& info = *reinterpret_cast<SVGMarkerLayoutInfo*>(infoPtr); SVGMarkerData& markerData = info.markerData(); int& elementIndex = info.elementIndex(); // First update the outslope for the previous element markerData.updateOutslope(element->points[0]); // Draw the marker for the previous element SVGResourceMarker* marker = markerData.marker(); if (elementIndex > 0 && marker) info.addLayoutedMarker(marker, markerData.origin(), markerData.currentAngle()); // Update our marker data for this element markerData.updateMarkerDataForPathElement(element); // After drawing the start marker, switch to drawing mid markers if (elementIndex == 1) markerData.updateTypeAndMarker(SVGMarkerData::Mid, info.midMarker()); ++elementIndex; } FloatRect SVGMarkerLayoutInfo::calculateBoundaries(SVGResourceMarker* startMarker, SVGResourceMarker* midMarker, SVGResourceMarker* endMarker, float strokeWidth, const Path& path) { m_layout.clear(); m_midMarker = midMarker; m_strokeWidth = strokeWidth; m_elementIndex = 0; m_markerData = SVGMarkerData(SVGMarkerData::Start, startMarker); path.apply(this, processStartAndMidMarkers); if (endMarker) { m_markerData.updateTypeAndMarker(SVGMarkerData::End, endMarker); addLayoutedMarker(endMarker, m_markerData.origin(), m_markerData.currentAngle()); } if (m_layout.isEmpty()) return FloatRect(); Vector<MarkerLayout>::iterator it = m_layout.begin(); Vector<MarkerLayout>::iterator end = m_layout.end(); FloatRect bounds; for (; it != end; ++it) { MarkerLayout& layout = *it; RenderSVGViewportContainer* markerContent = layout.marker->renderer(); ASSERT(markerContent); bounds.unite(markerContent->markerBoundaries(layout.matrix)); } return bounds; } void SVGMarkerLayoutInfo::drawMarkers(RenderObject::PaintInfo& paintInfo) { if (m_layout.isEmpty()) return; Vector<MarkerLayout>::iterator it = m_layout.begin(); Vector<MarkerLayout>::iterator end = m_layout.end(); for (; it != end; ++it) { MarkerLayout& layout = *it; layout.marker->draw(paintInfo, layout.matrix); } } void SVGMarkerLayoutInfo::addLayoutedMarker(SVGResourceMarker* marker, const FloatPoint& origin, float angle) { ASSERT(marker); m_layout.append(MarkerLayout(marker, marker->markerTransformation(origin, angle, m_strokeWidth))); } } #endif // ENABLE(SVG)