summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2007-05-15 05:43:53 +0000
committerJohn Wiegley <johnw@newartisans.com>2008-04-13 03:38:51 -0400
commitff43b1d135c3a1e43ed59d8d484e04f2ccffb3ee (patch)
treee2a6134051bc89a991e47076fef44ce9dfe378db
parent3244c693f8db9afb1845f5e6de3cb1807e68003d (diff)
downloadfork-ledger-ff43b1d135c3a1e43ed59d8d484e04f2ccffb3ee.tar.gz
fork-ledger-ff43b1d135c3a1e43ed59d8d484e04f2ccffb3ee.tar.bz2
fork-ledger-ff43b1d135c3a1e43ed59d8d484e04f2ccffb3ee.zip
Started working on an XPath visitor class
-rw-r--r--src/node.h2
-rw-r--r--src/value.cc3
-rw-r--r--src/xpath.cc2
-rw-r--r--src/xpath.h46
4 files changed, 48 insertions, 5 deletions
diff --git a/src/node.h b/src/node.h
index 5bc254ff..47edd10d 100644
--- a/src/node.h
+++ b/src/node.h
@@ -259,7 +259,7 @@ public:
}
virtual value_t to_value() const {
- return text();
+ return value_t(text(), true);
}
void print(std::ostream& out) const;
diff --git a/src/value.cc b/src/value.cc
index 3965dfbd..b3578ae9 100644
--- a/src/value.cc
+++ b/src/value.cc
@@ -1565,6 +1565,9 @@ void value_t::print(std::ostream& out, const int first_width,
case AMOUNT:
case STRING:
case POINTER:
+ // jww (2007-05-14): I need a version of this print just for XPath
+ // expression, since amounts and strings need to be output with
+ // special syntax.
out << *this;
break;
diff --git a/src/xpath.cc b/src/xpath.cc
index 6efa214a..4a20e402 100644
--- a/src/xpath.cc
+++ b/src/xpath.cc
@@ -2150,8 +2150,6 @@ void xpath_t::op_t::dump(std::ostream& out, const int depth) const
} else {
assert(! right());
}
- } else {
- assert(! left());
}
}
diff --git a/src/xpath.h b/src/xpath.h
index f17d3ee8..f1fc0dcf 100644
--- a/src/xpath.h
+++ b/src/xpath.h
@@ -231,7 +231,7 @@ public:
typedef std::list<element_t>::const_iterator element_iterator;
- struct node_appender_t {
+ struct value_node_appender_t {
value_t::sequence_t& sequence;
node_appender_t(value_t::sequence_t& _sequence)
: sequence(_sequence) {}
@@ -245,7 +245,7 @@ public:
void find_all(value_t::sequence_t& result,
node_t& start, scope_t * scope) {
- visit(start, scope, node_appender_t(result));
+ visit(start, scope, value_node_appender_t(result));
}
void visit(node_t& start, scope_t * scope,
@@ -261,6 +261,36 @@ public:
scope_t * scope, const function<void (node_t&)>& func);
};
+ class path_iterator_t
+ {
+ path_t path;
+ std::vector<node_t *> sequence;
+
+ struct node_appender_t {
+ std::vector<node_t *>& sequence;
+ node_appender_t(std::vector<node_t *>& _sequence)
+ : sequence(_sequence) {}
+ void operator()(node_t& node) {
+ sequence.push_back(&node);
+ }
+ };
+
+ public:
+ typedef std::vector<node_t *>::iterator iterator;
+ typedef std::vector<node_t *>::const_iterator const_iterator;
+
+ path_iterator_t(const xpath_t& path_expr, node_t& start, scope_t * scope)
+ : path(path_expr) {
+ path.visit(start, scope, node_appender_t(sequence));
+ }
+
+ iterator begin() { return sequence.begin(); }
+ const_iterator begin() const { return sequence.begin(); }
+
+ iterator end() { return sequence.end(); }
+ const_iterator end() const { return sequence.end(); }
+ };
+
struct op_t : public noncopyable
{
enum kind_t {
@@ -704,6 +734,18 @@ public:
return temp.calc(top, scope);
}
+ void find_all(value_t::sequence_t& result,
+ node_t& start, scope_t * scope) {
+ path_t path(*this);
+ path.find_all(result, start, scope);
+ }
+
+ void visit(node_t& start, scope_t * scope,
+ const function<void (node_t&)>& func) {
+ path_t path(*this);
+ path.visit(start, scope, func);
+ }
+
void print(std::ostream& out, xml::document_t& document) const {
print(out, document, true, NULL, NULL, NULL);
}