From b1b4e2aadff5983d443d70c09ea86a41b015873f Mon Sep 17 00:00:00 2001 From: John Wiegley Date: Sat, 12 Jun 2010 15:42:02 -0400 Subject: Add support for typed metadata The metadata construct 'Key: Value' is now just a special case for 'Key:: "Value"'. Another after a :: in metadata setting is parsed as a full value expression and typed as such. For example: ; Key:: $400 + $500 ledger -l 'tag("Key") < $1000' --- src/item.cc | 30 +++++++++++++++++++++++------- 1 file changed, 23 insertions(+), 7 deletions(-) (limited to 'src/item.cc') diff --git a/src/item.cc b/src/item.cc index 1f42510c..f59c9e29 100644 --- a/src/item.cc +++ b/src/item.cc @@ -135,7 +135,9 @@ item_t::set_tag(const string& tag, } } -void item_t::parse_tags(const char * p, bool overwrite_existing, +void item_t::parse_tags(const char * p, + scope_t& scope, + bool overwrite_existing, optional current_year) { if (const char * b = std::strchr(p, '[')) { @@ -164,14 +166,21 @@ void item_t::parse_tags(const char * p, bool overwrite_existing, std::strcpy(buf.get(), p); string tag; + bool by_value = false; for (char * q = std::strtok(buf.get(), " \t"); q; q = std::strtok(NULL, " \t")) { const string::size_type len = std::strlen(q); + if (len < 2) continue; if (! tag.empty()) { - string_map::iterator i = - set_tag(tag, string_value(string(p + (q - buf.get()))), - overwrite_existing); + string_map::iterator i; + string field(p + (q - buf.get())); + if (by_value) { + bind_scope_t bound_scope(scope, *this); + i = set_tag(tag, expr_t(field).calc(bound_scope), overwrite_existing); + } else { + i = set_tag(tag, string_value(field), overwrite_existing); + } (*i).second.second = true; break; } @@ -184,12 +193,19 @@ void item_t::parse_tags(const char * p, bool overwrite_existing, } } else if (q[len - 1] == ':') { // a metadata setting - tag = string(q, len - 1); + int index = 1; + if (q[len - 2] == ':') { + by_value = true; + index = 2; + } + tag = string(q, len - index); } } } -void item_t::append_note(const char * p, bool overwrite_existing, +void item_t::append_note(const char * p, + scope_t& scope, + bool overwrite_existing, optional current_year) { if (note) { @@ -199,7 +215,7 @@ void item_t::append_note(const char * p, bool overwrite_existing, note = p; } - parse_tags(p, overwrite_existing, current_year); + parse_tags(p, scope, overwrite_existing, current_year); } namespace { -- cgit v1.2.3