//////////////////////////////////////////////////////////////////////////////// // task - a command line task list manager. // // Copyright 2006 - 2009, Paul Beckingham. // All rights reserved. // // This program is free software; you can redistribute it and/or modify it under // the terms of the GNU General Public License as published by the Free Software // Foundation; either version 2 of the License, or (at your option) any later // version. // // This program is distributed in the hope that it will be useful, but WITHOUT // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS // FOR A PARTICULAR PURPOSE. See the GNU General Public License for more // details. // // You should have received a copy of the GNU General Public License along with // this program; if not, write to the // // Free Software Foundation, Inc., // 51 Franklin Street, Fifth Floor, // Boston, MA // 02110-1301 // USA // //////////////////////////////////////////////////////////////////////////////// #include // TODO Remove #include #include "Filter.h" #include "util.h" #include "main.h" //////////////////////////////////////////////////////////////////////////////// // For every Att in the filter, lookup the equivalent in Record, and perform a // match. Aren't filters easy now that everything is an attribute? bool Filter::pass (const Record& record) const { Record::const_iterator r; // First do description/annotation matches. foreach (att, (*this)) { // Descriptions have special handling. if (att->name () == "description") { if ((r = record.find (att->name ())) != record.end ()) { // A description match failure can be salvaged by an annotation match. if (! att->match (r->second)) { bool annoMatch = false; foreach (ra, record) { if (ra->first.length () > 11 && ra->first.substr (0, 11) == "annotation_") { if (att->match (ra->second)) { annoMatch = true; break; } } } if (!annoMatch) return false; } } else if (! att->match (Att ())) return false; } // Annotations are skipped. else if (att->name ().length () > 11 && att->name ().substr (0, 11) == "annotation_") { } else { // An individual attribute match failure is enough to fail the filter. if ((r = record.find (att->name ())) != record.end ()) { if (! att->match (r->second)) return false; } else if (! att->match (Att ())) return false; } } return true; } //////////////////////////////////////////////////////////////////////////////// void Filter::applySequence (std::vector& all, Sequence& sequence) { std::vector filtered; foreach (task, all) foreach (i, sequence) if (task->id == *i) filtered.push_back (*task); if (sequence.size () != filtered.size ()) { std::vector filteredSequence; foreach (task, filtered) filteredSequence.push_back (task->id); std::vector left; std::vector right; listDiff (filteredSequence, (std::vector &)sequence, left, right); if (left.size ()) throw std::string ("Sequence filtering error - please report this error"); if (right.size ()) { std::stringstream out; out << "Task"; if (right.size () > 1) out << "s"; foreach (r, right) out << " " << *r; out << " not found"; throw out.str (); } } all = filtered; } ////////////////////////////////////////////////////////////////////////////////