arabica/include/XSLT/impl/handler/xslt_key_handler.hpp

71 lines
2 KiB
C++

#ifndef ARABICA_XSLT_KEY_HANDLER_HPP
#define ARABICA_XSLT_KEY_HANDLER_HPP
#include "../xslt_key.hpp"
#include <XML/XMLCharacterClasses.hpp>
namespace Arabica
{
namespace XSLT
{
class KeyHandler : public SAX::DefaultHandler<std::string>
{
public:
KeyHandler(CompilationContext& context) :
context_(context),
key_(0)
{
} // KeyHandler
virtual void startElement(const std::string& /* namespaceURI */,
const std::string& /* localName */,
const std::string& qName,
const SAX::Attributes<std::string>& atts)
{
if(key_ != 0)
throw SAX::SAXException(qName + " can not contain elements");
static const ValueRule rules[] = { { "name", true, 0, 0 },
{ "match", true, 0, 0 },
{ "use", true, 0, 0 },
{ 0, false, 0, 0 } };
std::map<std::string, std::string> attrs = gatherAttributes(qName, atts, rules);
name_ = context_.processInternalQName(attrs["name"]).clarkName();
try
{
Key::MatchExprList matches = context_.xpath_match_no_variables(attrs["match"]);
Arabica::XPath::XPathExpression<std::string> use = context_.xpath_expression_no_variables(attrs["use"]);
key_ = new Key(matches, use);
} // try
catch(const Arabica::XPath::UnboundVariableException& uve)
{
throw SAX::SAXException("Variable references are not allowed in xsl:key match and use expressions");
} // catch
} // startElement
virtual void endElement(const std::string& /* namespaceURI */,
const std::string& /* localName */,
const std::string& /* qName */)
{
context_.stylesheet().add_key(name_, key_);
context_.pop();
} // endElement
virtual void characters(const std::string& ch)
{
verifyNoCharacterData(ch, "xsl:key");
} // characters
private:
CompilationContext& context_;
std::string name_;
Key* key_;
}; // class KeyHandler
} // namespace XSLT
} // namespace Arabica
#endif