arabica/include/XSLT/impl/xslt_sort.hpp

203 lines
5.5 KiB
C++
Executable file

#ifndef ARABICA_XSLT_SORT_HPP
#define ARABICA_XSLT_SORT_HPP
#include <algorithm>
#include "handler/xslt_value_validation.hpp"
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;
std::string datatype = datatype_->evaluateAsString(node, context_->xpathContext());
std::string order = order_->evaluateAsString(node, context_->xpathContext());
static const char* allowed_datatypes[] = { "text", "number", 0 };
static const char* allowed_orders[] = { "ascending", "descending", 0 };
validateValues("xsl:sort", "data-type", datatype, allowed_datatypes);
validateValues("xsl:sort", "order", order, allowed_orders);
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
{
double v1 = select_->evaluateAsNumber(n1, context_->xpathContext());
double v2 = select_->evaluateAsNumber(n2, context_->xpathContext());
bool nan1 = Arabica::XPath::isNaN(v1);
bool nan2 = Arabica::XPath::isNaN(v2);
if(((nan1 && nan2) || (v1 == v2)) && (sub_sort_))
return (*sub_sort_)(n1, n2);
if(nan2)
return false;
if(nan1)
return true;
return v1 < v2;
} // numberAscending
bool numberDescending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
{
double v1 = select_->evaluateAsNumber(n1, context_->xpathContext());
double v2 = select_->evaluateAsNumber(n2, context_->xpathContext());
bool nan1 = Arabica::XPath::isNaN(v1);
bool nan2 = Arabica::XPath::isNaN(v2);
if(((nan1 && nan2) || (v1 == v2)) && (sub_sort_))
return (*sub_sort_)(n1, n2);
if(nan1)
return false;
if(nan2)
return true;
return v1 > v2;
} // numberDescending
bool stringAscending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
{
std::string v1 = select_->evaluateAsString(n1, context_->xpathContext());
std::string v2 = select_->evaluateAsString(n2, context_->xpathContext());
if((v1 == v2) && (sub_sort_))
return (*sub_sort_)(n1, n2);
return v1 < v2;
} // stringAscending
bool stringDescending(const DOM::Node<std::string>& n1, const DOM::Node<std::string>& n2) const
{
std::string v1 = select_->evaluateAsString(n1, context_->xpathContext());
std::string v2 = select_->evaluateAsString(n2, context_->xpathContext());
if((v1 == v2) && (sub_sort_))
return (*sub_sort_)(n1, n2);
return v1 > v2;
} // stringAscending
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_;
ExecutionContext* context_;
Sort* sub_sort_;
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);
std::stable_sort(nodes.begin(), nodes.end(), SortP(*sort_));
} // 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