2012-11-02 21:01:15 +00:00
|
|
|
#ifndef ARABICA_XSLT_KEY_HANDLER_HPP
|
|
|
|
#define ARABICA_XSLT_KEY_HANDLER_HPP
|
|
|
|
|
|
|
|
#include "../xslt_key.hpp"
|
|
|
|
#include <XML/XMLCharacterClasses.hpp>
|
|
|
|
|
|
|
|
namespace Arabica
|
|
|
|
{
|
|
|
|
namespace XSLT
|
|
|
|
{
|
|
|
|
|
2012-11-06 19:02:31 +00:00
|
|
|
template<class string_type, class string_adaptor>
|
|
|
|
class KeyHandler : public SAX::DefaultHandler<string_type, string_adaptor>
|
2012-11-02 21:01:15 +00:00
|
|
|
{
|
2012-11-12 21:46:08 +00:00
|
|
|
typedef StylesheetConstant<string_type, string_adaptor> SC;
|
2012-11-15 22:03:42 +00:00
|
|
|
typedef AttributeValidators<string_type, string_adaptor> AV;
|
2012-11-06 19:02:31 +00:00
|
|
|
typedef typename Key<string_type, string_adaptor>::MatchExprList MatchExprList;
|
2012-11-02 21:01:15 +00:00
|
|
|
public:
|
2012-11-06 19:02:31 +00:00
|
|
|
KeyHandler(CompilationContext<string_type, string_adaptor>& context) :
|
2012-11-02 21:01:15 +00:00
|
|
|
context_(context),
|
|
|
|
key_(0)
|
|
|
|
{
|
|
|
|
} // KeyHandler
|
|
|
|
|
2012-11-06 19:02:31 +00:00
|
|
|
virtual void startElement(const string_type& /* namespaceURI */,
|
|
|
|
const string_type& /* localName */,
|
|
|
|
const string_type& qName,
|
|
|
|
const SAX::Attributes<string_type, string_adaptor>& atts)
|
2012-11-02 21:01:15 +00:00
|
|
|
{
|
|
|
|
if(key_ != 0)
|
2012-11-12 21:46:08 +00:00
|
|
|
throw SAX::SAXException(string_adaptor::asStdString(qName) + " can not contain elements");
|
2012-11-02 21:01:15 +00:00
|
|
|
|
2012-11-15 22:03:42 +00:00
|
|
|
static const AV rules = AV::rule(SC::name, true)
|
|
|
|
.rule(SC::match, true)
|
|
|
|
.rule(SC::use, true);
|
2012-11-02 21:01:15 +00:00
|
|
|
|
2012-11-15 22:03:42 +00:00
|
|
|
std::map<string_type, string_type> attrs = rules.gather(qName, atts);
|
2012-11-12 21:46:08 +00:00
|
|
|
name_ = context_.processInternalQName(attrs[SC::name]).clarkName();
|
2012-11-02 21:01:15 +00:00
|
|
|
try
|
|
|
|
{
|
2012-11-12 21:46:08 +00:00
|
|
|
MatchExprList matches = context_.xpath_match_no_variables(attrs[SC::match]);
|
|
|
|
Arabica::XPath::XPathExpression<string_type, string_adaptor> use = context_.xpath_expression_no_variables(attrs[SC::use]);
|
2012-11-02 21:01:15 +00:00
|
|
|
|
2012-11-06 19:02:31 +00:00
|
|
|
key_ = new Key<string_type, string_adaptor>(matches, use);
|
2012-11-02 21:01:15 +00:00
|
|
|
} // try
|
|
|
|
catch(const Arabica::XPath::UnboundVariableException&)
|
|
|
|
{
|
|
|
|
throw SAX::SAXException("Variable references are not allowed in xsl:key match and use expressions");
|
|
|
|
} // catch
|
|
|
|
} // startElement
|
|
|
|
|
2012-11-06 19:02:31 +00:00
|
|
|
virtual void endElement(const string_type& /* namespaceURI */,
|
2012-11-12 21:46:08 +00:00
|
|
|
const string_type& /* localName */,
|
|
|
|
const string_type& /* qName */)
|
2012-11-02 21:01:15 +00:00
|
|
|
{
|
|
|
|
context_.stylesheet().add_key(name_, key_);
|
|
|
|
context_.pop();
|
|
|
|
} // endElement
|
|
|
|
|
2012-11-06 19:02:31 +00:00
|
|
|
virtual void characters(const string_type& ch)
|
2012-11-02 21:01:15 +00:00
|
|
|
{
|
2012-11-12 21:46:08 +00:00
|
|
|
verifyNoCharacterData<string_type, string_adaptor>(ch, SC::key);
|
2012-11-02 21:01:15 +00:00
|
|
|
} // characters
|
|
|
|
|
|
|
|
private:
|
2012-11-06 19:02:31 +00:00
|
|
|
CompilationContext<string_type, string_adaptor>& context_;
|
|
|
|
string_type name_;
|
|
|
|
Key<string_type, string_adaptor>* key_;
|
2012-11-02 21:01:15 +00:00
|
|
|
}; // class KeyHandler
|
|
|
|
|
|
|
|
} // namespace XSLT
|
|
|
|
} // namespace Arabica
|
|
|
|
|
|
|
|
#endif
|