//////////////////////////////////////////////////////////////////////////////// // taskwarrior - a command line task list manager. // // Copyright 2006 - 2011, Paul Beckingham, Federico Hernandez. // 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 #include #include #include #include #include #include #include #include extern Context context; //////////////////////////////////////////////////////////////////////////////// CmdCustom::CmdCustom ( const std::string& k, const std::string& u, const std::string& d) { _keyword = k; _usage = u; _description = d; _read_only = true; _displays_id = true; } //////////////////////////////////////////////////////////////////////////////// int CmdCustom::execute (std::string& output) { int rc = 0; // Load report configuration. std::string reportColumns = context.config.get ("report." + _keyword + ".columns"); std::string reportLabels = context.config.get ("report." + _keyword + ".labels"); std::string reportSort = context.config.get ("report." + _keyword + ".sort"); std::string reportFilter = context.config.get ("report." + _keyword + ".filter"); std::vector columns; split (columns, reportColumns, ','); validateReportColumns (columns); std::vector labels; split (labels, reportLabels, ','); if (columns.size () != labels.size () && labels.size () != 0) throw std::string ("There are different numbers of columns and labels ") + "for report '" + _keyword + "'."; std::map columnLabels; if (labels.size ()) for (unsigned int i = 0; i < columns.size (); ++i) columnLabels[columns[i]] = labels[i]; std::vector sortOrder; split (sortOrder, reportSort, ','); validateSortColumns (sortOrder); // Prepend the argument list with those from the report filter. std::vector filterArgs; split (filterArgs, reportFilter, ' '); std::vector ::iterator arg; for (arg = filterArgs.begin (); arg != filterArgs.end (); ++ arg) { context.args.capture_first (*arg); context.a3.capture_first (*arg); } // Load the data. // TODO Replace with TDB2. std::vector tasks; context.tdb.lock (context.config.getBoolean ("locking")); handleRecurrence (); context.tdb.load (tasks); context.tdb.commit (); context.tdb.unlock (); // Apply filter. std::vector filtered; filter (tasks, filtered); // Sort the tasks. std::vector sequence; for (unsigned int i = 0; i < filtered.size (); ++i) sequence.push_back (i); sort_tasks (filtered, sequence, reportSort); // Configure the view. ViewTask view; view.width (context.getWidth ()); view.leftMargin (context.config.getInteger ("indent.report")); view.extraPadding (context.config.getInteger ("row.padding")); view.intraPadding (context.config.getInteger ("column.padding")); Color label (context.config.get ("color.label")); view.colorHeader (label); Color alternate (context.config.get ("color.alternate")); view.colorOdd (alternate); view.intraColorOdd (alternate); // Add the columns and labels. for (unsigned int i = 0; i < columns.size (); ++i) { Column* c = Column::factory (columns[i], _keyword); if (i < labels.size ()) c->setLabel (labels[i]); view.add (c); } // How many lines taken up by table header? // TODO Consider rc.verbose int table_header; if (context.color () && context.config.getBoolean ("fontunderline")) table_header = 1; // Underlining doesn't use extra line. else table_header = 2; // Dashes use an extra line. // Report output can be limited by rows or lines. int maxrows = 0; int maxlines = 0; getLimits (_keyword, maxrows, maxlines); // Adjust for fluff in the output. // TODO Consider rc.verbose if (maxlines) maxlines -= (context.verbose ("blank") ? 1 : 0) + table_header + context.footnotes.size () + 1; // "X tasks shown ..." // Render. // TODO Consider rc.verbose std::stringstream out; if (filtered.size ()) { view.truncateRows (maxrows); view.truncateLines (maxlines); out << optionalBlankLine () << view.render (filtered, sequence) << optionalBlankLine () << filtered.size () << (filtered.size () == 1 ? " task" : " tasks"); if (maxrows && maxrows < (int)filtered.size ()) out << ", " << maxrows << " shown"; if (maxlines && maxlines < (int)filtered.size ()) out << ", truncated to " << maxlines - table_header << " lines"; out << "\n"; } else { out << "No matches." << std::endl; rc = 1; } output = out.str (); return rc; } //////////////////////////////////////////////////////////////////////////////// void CmdCustom::validateReportColumns (std::vector & columns) { // One-time initialization, on demand. static std::map legacyMap; if (! legacyMap.size ()) { legacyMap["priority_long"] = "priority.long"; legacyMap["entry_time"] = "entry"; legacyMap["start_time"] = "start"; legacyMap["end_time"] = "end"; legacyMap["countdown"] = "due.countdown"; legacyMap["countdown_compact"] = "due.countdown"; legacyMap["age"] = "entry.age"; legacyMap["age_compact"] = "entry.age"; legacyMap["active"] = "start.active"; legacyMap["recurrence_indicator"] = "recur.indicator"; legacyMap["tag_indicator"] = "tags.indicator"; legacyMap["description_only"] = "description.desc"; } std::vector ::iterator i; for (i = columns.begin (); i != columns.end (); ++i) { // If a legacy column was used, complain about it, but modify it anyway. std::map ::iterator found = legacyMap.find (*i); if (found != legacyMap.end ()) { context.footnote (std::string ("Deprecated report field '") + *i + "' used. Please modify this to '" + found->second + "'."); *i = found->second; } } } //////////////////////////////////////////////////////////////////////////////// void CmdCustom::validateSortColumns (std::vector & columns) { // One-time initialization, on demand. static std::map legacyMap; if (! legacyMap.size ()) { legacyMap["priority_long"] = "priority"; legacyMap["entry_time"] = "entry"; legacyMap["start_time"] = "start"; legacyMap["end_time"] = "end"; legacyMap["countdown"] = "due"; legacyMap["countdown_compact"] = "due"; legacyMap["age"] = "entry"; legacyMap["age_compact"] = "entry"; legacyMap["active"] = "start"; legacyMap["recurrence_indicator"] = "recur"; legacyMap["tag_indicator"] = "tags"; legacyMap["description_only"] = "description"; } std::vector ::iterator i; for (i = columns.begin (); i != columns.end (); ++i) { // If a legacy column was used, complain about it, but modify it anyway. std::map ::iterator found = legacyMap.find (*i); if (found != legacyMap.end ()) { context.footnote (std::string ("Deprecated sort field '") + *i + "' used. Please modify this to '" + found->second + "'."); *i = found->second; } } } //////////////////////////////////////////////////////////////////////////////// // A value of zero mean unlimited. // A value of 'page' means however many screen lines there are. // A value of a positive integer is a row/task limit. void CmdCustom::getLimits (const std::string& report, int& rows, int& lines) { rows = 0; lines = 0; int screenheight = 0; // If a report has a stated limit, use it. if (report != "") { std::string name = "report." + report + ".limit"; if (context.config.get (name) == "page") lines = screenheight = context.getHeight (); else rows = context.config.getInteger (name); } // If the custom report has a defined limit, then allow a numeric override. // This is an integer specified as a filter (limit:10). std::string limit = context.args.find_limit (); if (limit != "") { if (limit == "page") { if (screenheight == 0) screenheight = context.getHeight (); rows = 0; lines = screenheight; } else { rows = (int) strtol (limit.c_str (), NULL, 10); lines = 0; } } } ////////////////////////////////////////////////////////////////////////////////