arabica/include/XSLT/impl/xslt_functions.hpp

229 lines
8.2 KiB
C++
Raw Normal View History

2007-07-19 19:01:42 +02:00
#ifndef ARABICA_XSLT_FUNCTIONS_HPP
#define ARABICA_XSLT_FUNCTIONS_HPP
2007-08-22 14:22:57 +02:00
2007-09-05 00:55:47 +02:00
#include <DOM/SAX2DOM/SAX2DOM.hpp>
#include <SAX/helpers/CatchErrorHandler.hpp>
2007-08-22 14:22:57 +02:00
namespace Arabica
{
namespace XSLT
{
// node-set document(object, node-set?)
class DocumentFunction : public Arabica::XPath::NodeSetXPathFunction<std::string>
2007-08-22 14:22:57 +02:00
{
typedef Arabica::XPath::NodeSetXPathFunction<std::string> baseT;
2007-08-22 14:22:57 +02:00
public:
2007-08-23 15:55:06 +02:00
DocumentFunction(const std::string& currentBase,
2007-10-25 22:42:00 +02:00
const std::vector<Arabica::XPath::XPathExpression<std::string> >& args) :
Arabica::XPath::NodeSetXPathFunction<std::string>(1, 2, args),
2007-08-23 15:55:06 +02:00
baseURI_(currentBase)
2007-08-22 14:22:57 +02:00
{ }
protected:
virtual Arabica::XPath::NodeSet<std::string> doEvaluate(const DOM::Node<std::string>& context,
const Arabica::XPath::ExecutionContext<std::string>& executionContext) const
2007-08-22 14:22:57 +02:00
{
2007-08-25 13:51:29 +02:00
Arabica::XPath::NodeSet<std::string> nodes;
Arabica::XPath::XPathValue<std::string> a0 = arg(0, context, executionContext);
if(argCount() != 1)
throw Arabica::XPath::UnsupportedException("two arg version of document()");
if(a0.type() != Arabica::XPath::STRING)
2007-08-25 13:51:29 +02:00
throw Arabica::XPath::UnsupportedException("node-set arg version of document()");
load_document(a0.asString(), nodes);
return nodes;
} // doEvaluate
2007-08-22 14:22:57 +02:00
private:
2007-08-25 13:51:29 +02:00
void load_document(const std::string& location,
Arabica::XPath::NodeSet<std::string>& nodes) const
2007-08-22 14:22:57 +02:00
{
SAX2DOM::Parser<std::string> domParser;
SAX::CatchErrorHandler<std::string> eh;
domParser.setErrorHandler(eh);
Arabica::io::URI base(baseURI_);
2007-09-28 10:30:16 +02:00
Arabica::io::URI absolute(base, location);
2007-08-23 15:55:06 +02:00
SAX::InputSource<std::string> is(absolute.as_string());
2007-08-22 14:22:57 +02:00
domParser.parse(is);
if(!eh.errorsReported())
2007-08-25 13:51:29 +02:00
nodes.push_back(domParser.getDocument());
2007-08-22 14:22:57 +02:00
else
std::cerr << eh.errors() << std::endl;
} // load_document
2007-08-23 15:55:06 +02:00
std::string baseURI_;
2007-08-22 14:22:57 +02:00
}; // DocumentFunction
// node-set key(string, object)
class KeyFunction : public Arabica::XPath::NodeSetXPathFunction<std::string>
{
typedef Arabica::XPath::NodeSetXPathFunction<std::string> baseT;
public:
KeyFunction(const DeclaredKeys& keys,
const std::map<std::string, std::string>& inscopeNamespaces,
const std::vector<Arabica::XPath::XPathExpression<std::string> >& args) :
Arabica::XPath::NodeSetXPathFunction<std::string>(2, 2, args),
keys_(keys),
namespaces_(inscopeNamespaces)
{
} // KeyFunction
protected:
virtual Arabica::XPath::NodeSet<std::string> doEvaluate(const DOM::Node<std::string>& context,
const Arabica::XPath::ExecutionContext<std::string>& executionContext) const
{
2009-03-03 08:58:29 +01:00
std::string keyname = argAsString(0, context, executionContext);
std::string keyClarkName = XML::QualifiedName<std::string>::parseQName(keyname, true, UriMapper(namespaces_)).clarkName();
Arabica::XPath::XPathValue<std::string> a1 = baseT::arg(1, context, executionContext);
if(a1.type() == Arabica::XPath::NODE_SET)
2009-03-03 08:58:29 +01:00
return nodeSetUnion(keyClarkName, a1.asNodeSet(), executionContext);
return keys_.lookup(keyClarkName, a1.asString(), executionContext);
} // doEvaluate
Arabica::XPath::NodeSet<std::string> nodeSetUnion(const std::string& keyClarkName,
2009-03-03 08:58:29 +01:00
const Arabica::XPath::NodeSet<std::string> nodes,
const Arabica::XPath::ExecutionContext<std::string>& executionContext) const
{
Arabica::XPath::NodeSet<std::string> results;
for(Arabica::XPath::NodeSet<std::string>::const_iterator n = nodes.begin(), ne = nodes.end(); n != ne; ++n)
{
std::string id = Arabica::XPath::impl::nodeStringValue<std::string, Arabica::default_string_adaptor<std::string> >(*n);
results.push_back(keys_.lookup(keyClarkName, id, executionContext));
} // for ...
results.to_document_order();
return results;
2009-03-03 08:58:29 +01:00
} // nodeSetUnion
private:
const DeclaredKeys& keys_;
std::map<std::string, std::string> namespaces_;
class UriMapper
{
public:
UriMapper(const std::map<std::string, std::string>& namespaces) : namespaces_(namespaces) { }
UriMapper(const UriMapper& rhs) : namespaces_(rhs.namespaces_) { }
std::string operator()(const std::string& prefix) const
{
std::map<std::string, std::string>::const_iterator ns = namespaces_.find(prefix);
if(ns == namespaces_.end())
return "";
return ns->second;
} //operator()
private:
const std::map<std::string, std::string>& namespaces_;
bool operator==(const UriMapper&) const;
UriMapper& operator=(const UriMapper&);
}; // class UriMapper
}; // class KeyFunction
// string format-number(number, string, string?)
// node-set current()
class CurrentFunction : public Arabica::XPath::NodeSetXPathFunction<std::string>
{
typedef Arabica::XPath::NodeSetXPathFunction<std::string> baseT;
public:
CurrentFunction(const std::vector<Arabica::XPath::XPathExpression<std::string> >& args) :
Arabica::XPath::NodeSetXPathFunction<std::string>(0, 0, args) { }
protected:
2010-01-10 23:02:43 +01:00
virtual Arabica::XPath::NodeSet<std::string> doEvaluate(const DOM::Node<std::string>& /* context */,
const Arabica::XPath::ExecutionContext<std::string>& executionContext) const
{
Arabica::XPath::NodeSet<std::string> set;
set.push_back(executionContext.currentNode());
return set;
} // doEvaluate
}; // CurrentFunction
// string unparsed-entity-uri(string)
class UnparsedEntityUriFunction : public Arabica::XPath::StringXPathFunction<std::string>
{
public:
UnparsedEntityUriFunction(const std::vector<Arabica::XPath::XPathExpression<std::string> >& args) :
Arabica::XPath::StringXPathFunction<std::string>(1, 1, args) { }
protected:
2010-01-10 23:02:43 +01:00
virtual std::string doEvaluate(const DOM::Node<std::string>& /* context */,
const Arabica::XPath::ExecutionContext<std::string>& /* executionContext */) const
{
// This is a minimal, but I think conformant, implementation
return "";
} // evaluate
}; // UnparsedEntityUri
// string generate-id(node-set?)
class GenerateIdFunction : public Arabica::XPath::StringXPathFunction<std::string>
2008-10-27 20:13:47 +01:00
{
typedef Arabica::XPath::StringXPathFunction<std::string> baseT;
2008-10-27 20:13:47 +01:00
public:
GenerateIdFunction(const std::vector<Arabica::XPath::XPathExpression<std::string> >& args) :
Arabica::XPath::StringXPathFunction<std::string>(0, 1, args) { }
2008-10-27 20:13:47 +01:00
protected:
virtual std::string doEvaluate(const DOM::Node<std::string>& context,
const Arabica::XPath::ExecutionContext<std::string>& executionContext) const
2008-10-27 20:13:47 +01:00
{
DOM::Node<std::string> node;
if(baseT::argCount() == 0)
node = context;
else
{
Arabica::XPath::NodeSet<std::string> ns = baseT::argAsNodeSet(0, context, executionContext);
if(ns.size() == 0)
return "";
2008-10-27 20:13:47 +01:00
node = ns.top();
} // if ...
std::ostringstream os;
os << node.underlying_impl();
return os.str();
} // doEvaluate
2008-10-27 20:13:47 +01:00
}; // class GenerateIdFunction
2007-08-22 14:22:57 +02:00
// object system-property(string)
class SystemPropertyFunction : public Arabica::XPath::StringXPathFunction<std::string>
2007-08-22 14:22:57 +02:00
{
typedef Arabica::XPath::StringXPathFunction<std::string> baseT;
2007-08-22 14:22:57 +02:00
public:
2007-10-25 22:42:00 +02:00
SystemPropertyFunction (const std::vector<Arabica::XPath::XPathExpression<std::string> >& args) :
Arabica::XPath::StringXPathFunction<std::string>(1, 1, args) { }
protected:
virtual std::string doEvaluate(const DOM::Node<std::string>& context,
const Arabica::XPath::ExecutionContext<std::string>& executionContext) const
2007-08-22 14:22:57 +02:00
{
std::string property = baseT::argAsString(0, context, executionContext);
std::string result;
if(property == "xsl:version")
return "1.0";
if(property == "xsl:vendor")
return "Jez Higgins, Jez UK Ltd";
2007-08-22 14:22:57 +02:00
else if(property == "xsl:vendor-url")
return "http://www.jezuk.co.uk/arabica";
return "";
2007-08-22 14:22:57 +02:00
} // evaluate
}; // SystemPropertyFunction
// boolean element-available(string)
// boolean function-available(string)
2007-07-19 19:01:42 +02:00
} // namespace XSLT
} // namespace Arabica
#endif // ARABICA_XSLT_FUNCTIONS_HPP
2007-08-22 14:22:57 +02:00