2007-07-19 17:01:42 +00:00
|
|
|
#ifndef ARABICA_XSLT_SORT_HPP
|
|
|
|
#define ARABICA_XSLT_SORT_HPP
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
namespace Arabica
|
|
|
|
{
|
|
|
|
namespace XSLT
|
|
|
|
{
|
|
|
|
|
|
|
|
class Sort
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Sort(const Arabica::XPath::XPathExpressionPtr<std::string>& select,
|
|
|
|
const Arabica::XPath::XPathExpressionPtr<std::string>& lang, //="language-code"
|
|
|
|
const Arabica::XPath::XPathExpressionPtr<std::string>& datatype, //="text|number|qname"
|
|
|
|
const Arabica::XPath::XPathExpressionPtr<std::string>& order, //="ascending|descending"
|
|
|
|
const Arabica::XPath::XPathExpressionPtr<std::string>& caseorder) : //="upper-first|lower-first
|
|
|
|
select_(select),
|
|
|
|
lang_(lang),
|
|
|
|
datatype_(datatype),
|
|
|
|
order_(order),
|
|
|
|
caseorder_(caseorder),
|
|
|
|
sub_sort_(0)
|
|
|
|
{
|
|
|
|
} // Sort
|
|
|
|
|
|
|
|
~Sort()
|
|
|
|
{
|
|
|
|
delete sub_sort_;
|
|
|
|
} // ~Sort
|
|
|
|
|
|
|
|
void set_context(const DOM::Node<std::string>& node, ExecutionContext& context)
|
|
|
|
{
|
|
|
|
context_ = &context;
|
2010-10-04 22:18:39 +01:00
|
|
|
const std::string datatype = datatype_->evaluateAsString(node, context_->xpathContext());
|
|
|
|
const std::string order = order_->evaluateAsString(node, context_->xpathContext());
|
|
|
|
const std::string caseorder = caseorder_->evaluateAsString(node, context_->xpathContext());
|
2007-07-19 17:01:42 +00:00
|
|
|
|
|
|
|
static const char* allowed_datatypes[] = { "text", "number", 0 };
|
|
|
|
static const char* allowed_orders[] = { "ascending", "descending", 0 };
|
2010-10-04 22:18:39 +01:00
|
|
|
static const char* allowed_case_orders[] = { "upper-first", "lower-first", 0 };
|
2007-07-19 17:01:42 +00:00
|
|
|
validateValues("xsl:sort", "data-type", datatype, allowed_datatypes);
|
|
|
|
validateValues("xsl:sort", "order", order, allowed_orders);
|
2010-10-04 22:18:39 +01:00
|
|
|
validateValues("xsl:sort", "case-order", caseorder, allowed_case_orders);
|
2007-07-19 17:01:42 +00:00
|
|
|
|
|
|
|
if(datatype == "number")
|
|
|
|
if(order == "ascending")
|
|
|
|
sort_fn_ = &Sort::numberAscending;
|
|
|
|
else
|
|
|
|
sort_fn_ = &Sort::numberDescending;
|
|
|
|
else
|
|
|
|
if(order == "ascending")
|
|
|
|
sort_fn_ = &Sort::stringAscending;
|
|
|
|
else
|
|
|
|
sort_fn_ = &Sort::stringDescending;
|
|
|
|
|
|
|
|
if(sub_sort_)
|
|
|
|
sub_sort_->set_context(node, context);
|
|
|
|
} // set_context
|
|
|
|
|
|
|
|
bool operator()(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
|
|
|
|
{
|
|
|
|
return (this->*sort_fn_)(n1, n2);
|
|
|
|
} // operator()
|
|
|
|
|
|
|
|
void add_sub_sort(Sort* sort)
|
|
|
|
{
|
|
|
|
if(!sub_sort_)
|
|
|
|
sub_sort_ = sort;
|
|
|
|
else
|
|
|
|
sub_sort_->add_sub_sort(sort);
|
|
|
|
} // add_sub_sort
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef bool(Sort::*sortFn)(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const;
|
|
|
|
bool numberAscending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
|
|
|
|
{
|
2010-02-16 08:59:45 +00:00
|
|
|
return numberSort(n1, n2, nanAscending, std::less<double>());
|
2007-07-19 17:01:42 +00:00
|
|
|
} // numberAscending
|
|
|
|
bool numberDescending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
|
2010-02-16 08:59:45 +00:00
|
|
|
{
|
|
|
|
return numberSort(n1, n2, nanDescending, std::greater<double>());
|
|
|
|
} // numberDescending
|
2010-02-20 09:27:32 +00:00
|
|
|
static bool nanAscending(bool, bool nan2)
|
2010-02-16 08:59:45 +00:00
|
|
|
{
|
|
|
|
return !nan2;
|
|
|
|
} // nanAscending
|
2010-02-20 09:27:32 +00:00
|
|
|
static bool nanDescending(bool nan1, bool)
|
2010-02-16 08:59:45 +00:00
|
|
|
{
|
|
|
|
return !nan1;
|
|
|
|
} // nanDescending
|
|
|
|
template<class NanCompare, class NumberCompare>
|
|
|
|
bool numberSort(const DOM::Node<std::string>& n1,
|
|
|
|
const DOM::Node<std::string>& n2,
|
|
|
|
NanCompare nanCompare,
|
|
|
|
NumberCompare compare) const
|
2007-07-19 17:01:42 +00:00
|
|
|
{
|
2008-11-03 22:19:59 +00:00
|
|
|
double v1 = grabAsNumber(n1);
|
|
|
|
double v2 = grabAsNumber(n2);
|
2007-07-19 17:01:42 +00:00
|
|
|
|
|
|
|
bool nan1 = Arabica::XPath::isNaN(v1);
|
|
|
|
bool nan2 = Arabica::XPath::isNaN(v2);
|
|
|
|
|
|
|
|
if(((nan1 && nan2) || (v1 == v2)) && (sub_sort_))
|
|
|
|
return (*sub_sort_)(n1, n2);
|
|
|
|
|
2010-02-16 08:59:45 +00:00
|
|
|
if(nan1 || nan2)
|
|
|
|
return nanCompare(nan1, nan2);
|
2007-07-19 17:01:42 +00:00
|
|
|
|
2010-02-16 08:59:45 +00:00
|
|
|
return compare(v1, v2);
|
|
|
|
} // numberSort
|
2007-07-19 17:01:42 +00:00
|
|
|
bool stringAscending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
|
|
|
|
{
|
2010-02-16 10:18:29 +00:00
|
|
|
return stringSort(n1, n2, std::less<std::string>());
|
2007-07-19 17:01:42 +00:00
|
|
|
} // stringAscending
|
|
|
|
bool stringDescending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
|
2010-02-16 10:18:29 +00:00
|
|
|
{
|
|
|
|
return stringSort(n1, n2, std::greater<std::string>());
|
|
|
|
} // stringAscending
|
|
|
|
template<class StringCompare>
|
|
|
|
bool stringSort(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2, StringCompare compare) const
|
2007-07-19 17:01:42 +00:00
|
|
|
{
|
2008-11-03 22:19:59 +00:00
|
|
|
std::string v1 = grabAsString(n1);
|
|
|
|
std::string v2 = grabAsString(n2);
|
2007-07-19 17:01:42 +00:00
|
|
|
|
|
|
|
if((v1 == v2) && (sub_sort_))
|
|
|
|
return (*sub_sort_)(n1, n2);
|
|
|
|
|
2010-02-16 10:18:29 +00:00
|
|
|
return compare(v1, v2);
|
|
|
|
} // stringSort
|
2007-07-19 17:01:42 +00:00
|
|
|
|
2008-11-03 22:19:59 +00:00
|
|
|
std::string grabAsString(const DOM::Node<std::string>& n) const
|
|
|
|
{
|
|
|
|
context_->setPosition(n, 1);
|
|
|
|
return select_->evaluateAsString(n, context_->xpathContext());
|
|
|
|
} // grabAsString
|
|
|
|
double grabAsNumber(const DOM::Node<std::string>& n) const
|
|
|
|
{
|
|
|
|
context_->setPosition(n, 1);
|
|
|
|
return select_->evaluateAsNumber(n, context_->xpathContext());
|
|
|
|
} // grabAsString
|
|
|
|
|
2007-07-19 17:01:42 +00:00
|
|
|
Arabica::XPath::XPathExpressionPtr<std::string> select_;
|
|
|
|
Arabica::XPath::XPathExpressionPtr<std::string> lang_;
|
|
|
|
Arabica::XPath::XPathExpressionPtr<std::string> datatype_;
|
|
|
|
Arabica::XPath::XPathExpressionPtr<std::string> order_;
|
|
|
|
Arabica::XPath::XPathExpressionPtr<std::string> caseorder_;
|
|
|
|
Sort* sub_sort_;
|
2010-01-11 09:02:17 +00:00
|
|
|
ExecutionContext* context_;
|
2007-07-19 17:01:42 +00:00
|
|
|
sortFn sort_fn_;
|
|
|
|
|
|
|
|
Sort& operator=(const Sort&);
|
|
|
|
bool operator==(const Sort&) const;
|
|
|
|
}; // class Sort
|
|
|
|
|
|
|
|
class Sortable
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
Sortable() :
|
|
|
|
sort_(0)
|
|
|
|
{
|
|
|
|
} // Sortable
|
|
|
|
|
|
|
|
~Sortable()
|
|
|
|
{
|
|
|
|
delete sort_;
|
|
|
|
} // ~Sortable
|
|
|
|
|
|
|
|
void sort(const DOM::Node<std::string>& node, Arabica::XPath::NodeSet<std::string>& nodes, ExecutionContext& context) const
|
|
|
|
{
|
|
|
|
if(!sort_)
|
|
|
|
{
|
|
|
|
if(!nodes.forward())
|
|
|
|
nodes.to_document_order();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sort_->set_context(node, context);
|
2007-11-03 01:55:20 +00:00
|
|
|
std::stable_sort(nodes.begin(), nodes.end(), SortP(*sort_));
|
2007-07-19 17:01:42 +00:00
|
|
|
} // sort
|
|
|
|
|
|
|
|
bool has_sort() const { return sort_ != 0; }
|
|
|
|
|
|
|
|
public:
|
|
|
|
void add_sort(Sort* sort)
|
|
|
|
{
|
|
|
|
if(!sort_)
|
|
|
|
sort_ = sort;
|
|
|
|
else
|
|
|
|
sort_->add_sub_sort(sort);
|
|
|
|
} // add_sort
|
|
|
|
|
|
|
|
private:
|
|
|
|
Sort* sort_;
|
|
|
|
|
|
|
|
struct SortP
|
|
|
|
{
|
|
|
|
SortP(Sort& sort) : sort_(sort) { }
|
|
|
|
bool operator()(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2)
|
|
|
|
{
|
|
|
|
return sort_(n1, n2);
|
|
|
|
} // operator()
|
|
|
|
|
|
|
|
private:
|
|
|
|
Sort& sort_;
|
|
|
|
}; // struct SortP
|
|
|
|
}; // class Sortable
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace XSLT
|
|
|
|
} // namespace Arabica
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|