blob: 16d9bcb4208b8b239f19074de480e8464ab74498 [file] [log] [blame]
/**************************************************************
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_svgio.hxx"
#include <svgio/svgreader/svgnode.hxx>
#include <basegfx/polygon/b2dpolypolygontools.hxx>
#include <svgio/svgreader/svgdocument.hxx>
#include <svgio/svgreader/svgnode.hxx>
#include <svgio/svgreader/svgstyleattributes.hxx>
#include <drawinglayer/primitive2d/objectinfoprimitive2d.hxx>
#include <tools/urlobj.hxx>
//////////////////////////////////////////////////////////////////////////////
namespace svgio
{
namespace svgreader
{
/// #125258#
bool SvgNode::supportsParentStyle() const
{
return true;
}
const SvgStyleAttributes* SvgNode::getSvgStyleAttributes() const
{
return 0;
}
const SvgStyleAttributes* SvgNode::checkForCssStyle(const rtl::OUString& rClassStr, const SvgStyleAttributes& rOriginal) const
{
if(maCssStyleVector.empty()) // #120435# Evaluate for CSS styles only once, this cannot change
{
const SvgDocument& rDocument = getDocument();
if(rDocument.hasSvgStyleAttributesById())
{
if(getClass())
{
// find all referenced CSS styles, a list of entries is allowed
const rtl::OUString* pClassList = getClass();
const sal_Int32 nLen(pClassList->getLength());
sal_Int32 nPos(0);
const SvgStyleAttributes* pNew = 0;
skip_char(*pClassList, sal_Unicode(' '), nPos, nLen);
while(nPos < nLen)
{
rtl::OUStringBuffer aTokenValue;
copyToLimiter(*pClassList, sal_Unicode(' '), nPos, aTokenValue, nLen);
skip_char(*pClassList, sal_Unicode(' '), nPos, nLen);
rtl::OUString aId(rtl::OUString::createFromAscii("."));
const rtl::OUString aOUTokenValue(aTokenValue.makeStringAndClear());
// look for CSS style common to token
aId = aId + aOUTokenValue;
pNew = rDocument.findSvgStyleAttributesById(aId);
if(!pNew && rClassStr.getLength())
{
// look for CSS style common to class.token
aId = rClassStr + aId;
pNew = rDocument.findSvgStyleAttributesById(aId);
}
if(pNew)
{
const_cast< SvgNode* >(this)->maCssStyleVector.push_back(pNew);
}
}
}
if(maCssStyleVector.empty() && getId())
{
// if none found, search for CSS style equal to Id
const SvgStyleAttributes* pNew = rDocument.findSvgStyleAttributesById(*getId());
if(pNew)
{
const_cast< SvgNode* >(this)->maCssStyleVector.push_back(pNew);
}
}
if(maCssStyleVector.empty() && rClassStr.getLength())
{
// if none found, search for CSS style equal to class type
const SvgStyleAttributes* pNew = rDocument.findSvgStyleAttributesById(rClassStr);
if(pNew)
{
const_cast< SvgNode* >(this)->maCssStyleVector.push_back(pNew);
}
}
}
}
if(!maCssStyleVector.empty())
{
// #123510# if CSS styles were found, create a linked list with rOriginal as parent
// and all CSS styles as linked children, so that the style attribute has
// priority over the CSS style. If there is no style attribute this means that
// no values are set at rOriginal, thus it is still correct to have that order.
// Repeated style requests should only be issued from sub-Text nodes and I'm not
// sure if in-between text nodes may build other chains (should not happen). But
// it's only a re-chaining with pointers (cheap), so allow to do it every time.
SvgStyleAttributes* pCurrent = const_cast< SvgStyleAttributes* >(&rOriginal);
pCurrent->setCssStyleParent(0);
for(sal_uInt32 a(0); a < maCssStyleVector.size(); a++)
{
SvgStyleAttributes* pNext = const_cast< SvgStyleAttributes* >(maCssStyleVector[a]);
pCurrent->setCssStyleParent(pNext);
pCurrent = pNext;
pCurrent->setCssStyleParent(0);
}
}
return &rOriginal;
}
SvgNode::SvgNode(
SVGToken aType,
SvgDocument& rDocument,
SvgNode* pParent)
: maType(aType),
mrDocument(rDocument),
mpParent(pParent),
mpAlternativeParent(0),
maChildren(),
mpId(0),
mpClass(0),
maXmlSpace(XmlSpace_notset),
maDisplay(Display_inline),
maCssStyleVector()
{
OSL_ENSURE(SVGTokenUnknown != maType, "SvgNode with unknown type created (!)");
if(pParent)
{
pParent->maChildren.push_back(this);
}
else
{
#ifdef DBG_UTIL
if(SVGTokenSvg != getType())
{
OSL_ENSURE(false, "No parent for this node (!)");
}
#endif
}
}
SvgNode::~SvgNode()
{
while(maChildren.size())
{
delete maChildren[maChildren.size() - 1];
maChildren.pop_back();
}
if(mpId) delete mpId;
if(mpClass) delete mpClass;
}
void SvgNode::parseAttributes(const com::sun::star::uno::Reference< com::sun::star::xml::sax::XAttributeList >& xAttribs)
{
const sal_uInt32 nAttributes(xAttribs->getLength());
// #122522# SVG defines that 'In general, this means that the presentation attributes have
// lower priority than other CSS style rules specified in author style sheets or ‘style’
// attributes.' in http://www.w3.org/TR/SVG/styling.html#UsingPresentationAttributes
// (6.4 Specifying properties using the presentation attributes SVG 1.1). That means that
// e.g. font-size will appear as presentation attribute and CSS style attribute. In these
// cases, CSS style attributes need to have precedence. To do so it is possible to create
// a proirity system for all properties of a shape, but it will also work to parse the
// presentation attributes of type 'style' last, so they will overwrite the less-prioritized
// already interpreted ones. Thus, remember SVGTokenStyle entries and parse them last.
// To make this work it is required that parseAttribute is only called by parseAttributes
// which is the case.
std::vector< sal_uInt32 > aSVGTokenStyleIndexes;
for(sal_uInt32 a(0); a < nAttributes; a++)
{
const ::rtl::OUString aTokenName(xAttribs->getNameByIndex(a));
const SVGToken aSVGToken(StrToSVGToken(aTokenName));
if(SVGTokenStyle == aSVGToken)
{
// #122522# remember SVGTokenStyle entry
aSVGTokenStyleIndexes.push_back(a);
}
else
{
parseAttribute(aTokenName, aSVGToken, xAttribs->getValueByIndex(a));
}
}
// #122522# parse SVGTokenStyle entries last to override already interpreted
// 'presentation attributes' of potenially the same type
for(sal_uInt32 b(0); b < aSVGTokenStyleIndexes.size(); b++)
{
const sal_uInt32 nSVGTokenStyleIndex(aSVGTokenStyleIndexes[b]);
const ::rtl::OUString aTokenName(xAttribs->getNameByIndex(nSVGTokenStyleIndex));
parseAttribute(aTokenName, SVGTokenStyle, xAttribs->getValueByIndex(nSVGTokenStyleIndex));
}
}
Display getDisplayFromContent(const rtl::OUString& aContent)
{
if(aContent.getLength())
{
static rtl::OUString aStrInline(rtl::OUString::createFromAscii("inline"));
static rtl::OUString aStrBlock(rtl::OUString::createFromAscii("block"));
static rtl::OUString aStrList_item(rtl::OUString::createFromAscii("list-item"));
static rtl::OUString aStrRun_in(rtl::OUString::createFromAscii("run-in"));
static rtl::OUString aStrCompact(rtl::OUString::createFromAscii("compact"));
static rtl::OUString aStrMarker(rtl::OUString::createFromAscii("marker"));
static rtl::OUString aStrTable(rtl::OUString::createFromAscii("table"));
static rtl::OUString aStrInline_table(rtl::OUString::createFromAscii("inline-table"));
static rtl::OUString aStrTable_row_group(rtl::OUString::createFromAscii("table-row-group"));
static rtl::OUString aStrTable_header_group(rtl::OUString::createFromAscii("table-header-group"));
static rtl::OUString aStrTable_footer_group(rtl::OUString::createFromAscii("table-footer-group"));
static rtl::OUString aStrTable_row(rtl::OUString::createFromAscii("table-row"));
static rtl::OUString aStrTable_column_group(rtl::OUString::createFromAscii("table-column-group"));
static rtl::OUString aStrTable_column(rtl::OUString::createFromAscii("table-column"));
static rtl::OUString aStrTable_cell(rtl::OUString::createFromAscii("table-cell"));
static rtl::OUString aStrTable_caption(rtl::OUString::createFromAscii("table-caption"));
static rtl::OUString aStrNone(rtl::OUString::createFromAscii("none"));
static rtl::OUString aStrInherit(rtl::OUString::createFromAscii("inherit"));
if(aContent.match(aStrInline))
{
return Display_inline;
}
else if(aContent.match(aStrNone))
{
return Display_none;
}
else if(aContent.match(aStrInherit))
{
return Display_inherit;
}
else if(aContent.match(aStrBlock))
{
return Display_block;
}
else if(aContent.match(aStrList_item))
{
return Display_list_item;
}
else if(aContent.match(aStrRun_in))
{
return Display_run_in;
}
else if(aContent.match(aStrCompact))
{
return Display_compact;
}
else if(aContent.match(aStrMarker))
{
return Display_marker;
}
else if(aContent.match(aStrTable))
{
return Display_table;
}
else if(aContent.match(aStrInline_table))
{
return Display_inline_table;
}
else if(aContent.match(aStrTable_row_group))
{
return Display_table_row_group;
}
else if(aContent.match(aStrTable_header_group))
{
return Display_table_header_group;
}
else if(aContent.match(aStrTable_footer_group))
{
return Display_table_footer_group;
}
else if(aContent.match(aStrTable_row))
{
return Display_table_row;
}
else if(aContent.match(aStrTable_column_group))
{
return Display_table_column_group;
}
else if(aContent.match(aStrTable_column))
{
return Display_table_column;
}
else if(aContent.match(aStrTable_cell))
{
return Display_table_cell;
}
else if(aContent.match(aStrTable_caption))
{
return Display_table_caption;
}
}
// return the default
return Display_inline;
}
void SvgNode::parseAttribute(const rtl::OUString& /*rTokenName*/, SVGToken aSVGToken, const rtl::OUString& aContent)
{
switch(aSVGToken)
{
case SVGTokenId:
{
if(aContent.getLength())
{
setId(&aContent);
}
break;
}
case SVGTokenClass:
{
if(aContent.getLength())
{
setClass(&aContent);
}
break;
}
case SVGTokenXmlSpace:
{
if(aContent.getLength())
{
static rtl::OUString aStrDefault(rtl::OUString::createFromAscii("default"));
static rtl::OUString aStrPreserve(rtl::OUString::createFromAscii("preserve"));
if(aContent.match(aStrDefault))
{
setXmlSpace(XmlSpace_default);
}
else if(aContent.match(aStrPreserve))
{
setXmlSpace(XmlSpace_preserve);
}
}
break;
}
case SVGTokenDisplay:
{
if(aContent.getLength())
{
setDisplay(getDisplayFromContent(aContent));
}
break;
}
default:
{
break;
}
}
}
void SvgNode::decomposeSvgNode(drawinglayer::primitive2d::Primitive2DSequence& rTarget, bool bReferenced) const
{
if(Display_none == getDisplay())
{
return;
}
if(!bReferenced)
{
if(SVGTokenDefs == getType() ||
SVGTokenSymbol == getType() ||
SVGTokenClipPathNode == getType() ||
SVGTokenMask == getType() ||
SVGTokenMarker == getType() ||
SVGTokenPattern == getType())
{
// do not decompose defs or symbol nodes (these hold only style-like
// objects which may be used by referencing them) except when doing
// so controlled referenced
// also do not decompose ClipPaths and Masks. These should be embedded
// in a defs node (which gets not decomposed by itself), but you never
// know
// also not directly used are Markers and Patterns, only indirecty used
// by reference
// #121656# also do not decompose nodes which have display="none" set
// as property
return;
}
}
const SvgNodeVector& rChildren = getChildren();
if(!rChildren.empty())
{
const sal_uInt32 nCount(rChildren.size());
for(sal_uInt32 a(0); a < nCount; a++)
{
SvgNode* pCandidate = rChildren[a];
if(pCandidate && Display_none != pCandidate->getDisplay())
{
drawinglayer::primitive2d::Primitive2DSequence aNewTarget;
pCandidate->decomposeSvgNode(aNewTarget, bReferenced);
if(aNewTarget.hasElements())
{
drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(rTarget, aNewTarget);
}
}
else
{
OSL_ENSURE(false, "Null-Pointer in child node list (!)");
}
}
if(rTarget.hasElements())
{
const SvgStyleAttributes* pStyles = getSvgStyleAttributes();
if(pStyles)
{
// check if we have Title or Desc
const rtl::OUString& rTitle = pStyles->getTitle();
const rtl::OUString& rDesc = pStyles->getDesc();
if(rTitle.getLength() || rDesc.getLength())
{
// default object name is empty
rtl::OUString aObjectName;
// use path as object name when outmost element
if(SVGTokenSvg == getType())
{
aObjectName = getDocument().getAbsolutePath();
if(aObjectName.getLength())
{
INetURLObject aURL(aObjectName);
aObjectName = aURL.getName(
INetURLObject::LAST_SEGMENT,
true,
INetURLObject::DECODE_WITH_CHARSET);
}
}
// pack in ObjectInfoPrimitive2D group
const drawinglayer::primitive2d::Primitive2DReference xRef(
new drawinglayer::primitive2d::ObjectInfoPrimitive2D(
rTarget,
aObjectName,
rTitle,
rDesc));
rTarget = drawinglayer::primitive2d::Primitive2DSequence(&xRef, 1);
}
}
}
}
}
const basegfx::B2DRange SvgNode::getCurrentViewPort() const
{
if(getParent())
{
return getParent()->getCurrentViewPort();
}
else
{
return basegfx::B2DRange(); // return empty B2DRange
}
}
double SvgNode::getCurrentFontSize() const
{
if(getSvgStyleAttributes())
{
return getSvgStyleAttributes()->getFontSize().solve(*this, xcoordinate);
}
else if(getParent())
{
return getParent()->getCurrentFontSize();
}
else
{
return 0.0;
}
}
double SvgNode::getCurrentXHeight() const
{
if(getSvgStyleAttributes())
{
// for XHeight, use FontSize currently
return getSvgStyleAttributes()->getFontSize().solve(*this, ycoordinate);
}
else if(getParent())
{
return getParent()->getCurrentXHeight();
}
else
{
return 0.0;
}
}
void SvgNode::setId(const rtl::OUString* pfId)
{
if(mpId)
{
mrDocument.removeSvgNodeFromMapper(*mpId);
delete mpId;
mpId = 0;
}
if(pfId)
{
mpId = new rtl::OUString(*pfId);
mrDocument.addSvgNodeToMapper(*mpId, *this);
}
}
void SvgNode::setClass(const rtl::OUString* pfClass)
{
if(mpClass)
{
mrDocument.removeSvgNodeFromMapper(*mpClass);
delete mpClass;
mpClass = 0;
}
if(pfClass)
{
mpClass = new rtl::OUString(*pfClass);
mrDocument.addSvgNodeToMapper(*mpClass, *this);
}
}
XmlSpace SvgNode::getXmlSpace() const
{
if(maXmlSpace != XmlSpace_notset)
{
return maXmlSpace;
}
if(getParent())
{
return getParent()->getXmlSpace();
}
// default is XmlSpace_default
return XmlSpace_default;
}
} // end of namespace svgreader
} // end of namespace svgio
//////////////////////////////////////////////////////////////////////////////
// eof