blob: 65a214abeb6c861e9ea81de3f8bb55ef9b35dd1d [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.
*/
#include "ElemTemplate.hpp"
#include <xercesc/sax/AttributeList.hpp>
#include <xalanc/PlatformSupport/DOMStringHelper.hpp>
#include <xalanc/PlatformSupport/DoubleSupport.hpp>
#include <xalanc/PlatformSupport/XalanMessageLoader.hpp>
#include <xalanc/XPath/XPath.hpp>
#include <xalanc/XPath/XalanQNameByValue.hpp>
#include "Constants.hpp"
#include "Stylesheet.hpp"
#include "StylesheetConstructionContext.hpp"
namespace XALAN_CPP_NAMESPACE {
static const XalanQNameByValue s_empty(XalanMemMgrs::getDummyMemMgr());
ElemTemplate::ElemTemplate(
StylesheetConstructionContext& constructionContext,
Stylesheet& stylesheetTree,
const AttributeListType& atts,
XalanFileLoc lineNumber,
XalanFileLoc columnNumber) :
ElemTemplateElement(
constructionContext,
stylesheetTree,
lineNumber,
columnNumber,
StylesheetConstructionContext::ELEMNAME_TEMPLATE),
m_matchPattern(0),
m_name(&s_empty),
m_mode(&s_empty),
m_priority(XPath::getMatchScoreValue(XPath::eMatchScoreNone))
{
const XalanSize_t nAttrs = atts.getLength();
for (XalanSize_t i = 0; i < nAttrs; i++)
{
const XalanDOMChar* const aname = atts.getName(i);
if (equals(aname, Constants::ATTRNAME_MATCH))
{
m_matchPattern = constructionContext.createMatchPattern(getLocator(), atts.getValue(i), *this);
}
else if (equals(aname, Constants::ATTRNAME_NAME))
{
m_name = constructionContext.createXalanQName(
atts.getValue(i),
getStylesheet().getNamespaces(),
getLocator());
if (m_name->isValid() == false)
{
error(
constructionContext,
XalanMessages::AttributeValueNotValidQName_2Param,
aname,
atts.getValue(i));
}
}
else if (equals(aname, Constants::ATTRNAME_PRIORITY))
{
assert(atts.getValue(i) != 0);
m_priority = DoubleSupport::toDouble(atts.getValue(i), constructionContext.getMemoryManager());
}
else if (equals(aname, Constants::ATTRNAME_MODE))
{
m_mode = constructionContext.createXalanQName(
atts.getValue(i),
getStylesheet().getNamespaces(),
getLocator());
if (m_mode->isValid() == false)
{
error(
constructionContext,
XalanMessages::AttributeValueNotValidQName_2Param,
aname,
atts.getValue(i));
}
}
else if(isAttrOK(
aname,
atts,
i,
constructionContext) == false &&
processSpaceAttr(
Constants::ELEMNAME_TEMPLATE_WITH_PREFIX_STRING.c_str(),
aname,
atts,
i,
constructionContext) == false)
{
error(
constructionContext,
XalanMessages::ElementHasIllegalAttribute_2Param,
Constants::ELEMNAME_TEMPLATE_WITH_PREFIX_STRING.c_str(),
aname);
}
}
if (0 == m_matchPattern && m_name->isEmpty() == true)
{
error(
constructionContext,
XalanMessages::ElementRequiresEitherNameOrMatchAttribute_1Param,
Constants::ELEMNAME_TEMPLATE_WITH_PREFIX_STRING);
}
assert(m_name->isEmpty() == true || m_name->isValid() == true);
assert(m_mode->isEmpty() == true || m_mode->isValid() == true);
}
ElemTemplate::~ElemTemplate()
{
}
void
ElemTemplate::addToStylesheet(
StylesheetConstructionContext& constructionContext,
Stylesheet& theStylesheet)
{
theStylesheet.addTemplate(this, constructionContext);
}
const XalanQName&
ElemTemplate::getNameAttribute() const
{
return *m_name;
}
const XalanDOMString&
ElemTemplate::getElementName() const
{
return Constants::ELEMNAME_TEMPLATE_WITH_PREFIX_STRING;
}
#if !defined(XALAN_RECURSIVE_STYLESHEET_EXECUTION)
const ElemTemplateElement*
ElemTemplate::startElement(StylesheetExecutionContext& executionContext) const
{
ParentType::startElement(executionContext);
executionContext.pushCurrentTemplate(this);
return beginExecuteChildren(executionContext);
}
void
ElemTemplate::endElement(StylesheetExecutionContext& executionContext) const
{
executionContext.popCurrentTemplate();
endExecuteChildren(executionContext);
}
const ElemTemplateElement*
ElemTemplate::getInvoker(StylesheetExecutionContext& executionContext) const
{
return executionContext.getInvoker();
}
#endif
#if defined(XALAN_RECURSIVE_STYLESHEET_EXECUTION)
void
ElemTemplate::execute(StylesheetExecutionContext& executionContext) const
{
ParentType::execute(executionContext);
executeChildren(executionContext);
}
#endif
const XPath*
ElemTemplate::getXPath(XalanSize_t index) const
{
return index == 0 ? m_matchPattern : 0;
}
bool
ElemTemplate::childTypeAllowed(int xslToken) const
{
bool fResult = true;
switch(xslToken)
{
case StylesheetConstructionContext::ELEMNAME_WITH_PARAM:
fResult = false;
break;
default:
break;
}
return fResult;
}
#if defined(XALAN_RECURSIVE_STYLESHEET_EXECUTION)
void
ElemTemplate::executeChildren(StylesheetExecutionContext& executionContext) const
{
StylesheetExecutionContext::PushAndPopCurrentTemplate thePushAndPop(executionContext, this);
ParentType::executeChildren(executionContext);
}
void
ElemTemplate::executeChildren(
StylesheetExecutionContext& executionContext,
XalanNode* sourceNode) const
{
ParentType::executeChildren(executionContext, sourceNode);
}
#endif
}