blob: 7518333050dc2c2b71cd6c27054b915d15e1cb8d [file]
/*
* Copyright (C) 2007, 2010 Rob Buis <buis@kde.org>
* Copyright (C) 2018-2019 Apple Inc. All rights reserved.
*
* 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 "config.h"
#include "SVGViewSpec.h"
#include "Document.h"
#include "SVGElement.h"
#include "SVGFitToViewBox.h"
#include "SVGNames.h"
#include "SVGParserUtilities.h"
#include "SVGTransformList.h"
#include "SVGTransformable.h"
#include <wtf/TZoneMallocInlines.h>
#include <wtf/text/StringParsingBuffer.h>
namespace WebCore {
WTF_MAKE_TZONE_ALLOCATED_IMPL(SVGViewSpec);
SVGViewSpec::SVGViewSpec(SVGElement& contextElement)
: SVGFitToViewBox(&contextElement, SVGPropertyAccess::ReadOnly)
, m_contextElement(contextElement)
, m_transform(SVGTransformList::create(&contextElement, SVGPropertyAccess::ReadOnly))
{
static std::once_flag onceFlag;
std::call_once(onceFlag, [] {
PropertyRegistry::registerProperty<SVGNames::transformAttr, &SVGViewSpec::m_transform>();
});
}
RefPtr<SVGElement> SVGViewSpec::viewTarget() const
{
RefPtr contextElement = m_contextElement.get();
if (!contextElement)
return nullptr;
return dynamicDowncast<SVGElement>(contextElement->treeScope().getElementById(m_viewTargetString));
}
Ref<SVGTransformList> SVGViewSpec::protectedTransform()
{
return m_transform;
}
void SVGViewSpec::reset()
{
m_viewTargetString = emptyString();
protectedTransform()->clearItems();
SVGFitToViewBox::reset();
SVGZoomAndPan::reset();
}
template<typename CharacterType> static constexpr std::array<CharacterType, 7> svgViewSpec { 's', 'v', 'g', 'V', 'i', 'e', 'w' };
template<typename CharacterType> static constexpr std::array<CharacterType, 7> viewBoxSpec { 'v', 'i', 'e', 'w', 'B', 'o', 'x' };
template<typename CharacterType> static constexpr std::array<CharacterType, 19> preserveAspectRatioSpec { 'p', 'r', 'e', 's', 'e', 'r', 'v', 'e', 'A', 's', 'p', 'e', 'c', 't', 'R', 'a', 't', 'i', 'o' };
template<typename CharacterType> static constexpr std::array<CharacterType, 9> transformSpec { 't', 'r', 'a', 'n', 's', 'f', 'o', 'r', 'm' };
template<typename CharacterType> static constexpr std::array<CharacterType, 10> zoomAndPanSpec { 'z', 'o', 'o', 'm', 'A', 'n', 'd', 'P', 'a', 'n' };
template<typename CharacterType> static constexpr std::array<CharacterType, 10> viewTargetSpec { 'v', 'i', 'e', 'w', 'T', 'a', 'r', 'g', 'e', 't' };
bool SVGViewSpec::parseViewSpec(StringView string)
{
return readCharactersForParsing(string, [&]<typename CharacterType> (StringParsingBuffer<CharacterType> buffer) -> bool {
if (buffer.atEnd() || !m_contextElement)
return false;
if (!skipCharactersExactly(buffer, std::span { svgViewSpec<CharacterType> }))
return false;
if (!skipExactly(buffer, '('))
return false;
while (buffer.hasCharactersRemaining() && *buffer != ')') {
if (*buffer == 'v') {
if (skipCharactersExactly(buffer, std::span { viewBoxSpec<CharacterType> })) {
if (!skipExactly(buffer, '('))
return false;
auto viewBox = SVGFitToViewBox::parseViewBox(buffer, false);
if (!viewBox)
return false;
setViewBox(WTFMove(*viewBox));
if (!skipExactly(buffer, ')'))
return false;
} else if (skipCharactersExactly(buffer, std::span { viewTargetSpec<CharacterType> })) {
if (!skipExactly(buffer, '('))
return false;
auto viewTargetStart = buffer.position();
skipUntil(buffer, ')');
if (buffer.atEnd())
return false;
m_viewTargetString = String({ viewTargetStart, buffer.position() });
++buffer;
} else
return false;
} else if (*buffer == 'z') {
if (!skipCharactersExactly(buffer, std::span { zoomAndPanSpec<CharacterType> }))
return false;
if (!skipExactly(buffer, '('))
return false;
auto zoomAndPan = SVGZoomAndPan::parseZoomAndPan(buffer);
if (!zoomAndPan)
return false;
setZoomAndPan(*zoomAndPan);
if (!skipExactly(buffer, ')'))
return false;
} else if (*buffer == 'p') {
if (!skipCharactersExactly(buffer, std::span { preserveAspectRatioSpec<CharacterType> }))
return false;
if (!skipExactly(buffer, '('))
return false;
SVGPreserveAspectRatioValue preserveAspectRatio;
if (!preserveAspectRatio.parse(buffer, false))
return false;
setPreserveAspectRatio(preserveAspectRatio);
if (!skipExactly(buffer, ')'))
return false;
} else if (*buffer == 't') {
if (!skipCharactersExactly(buffer, std::span { transformSpec<CharacterType> }))
return false;
if (!skipExactly(buffer, '('))
return false;
protectedTransform()->parse(buffer);
if (!skipExactly(buffer, ')'))
return false;
} else
return false;
skipExactly(buffer, ';');
}
if (buffer.atEnd() || *buffer != ')')
return false;
return true;
});
}
}