//////////////////////////////////////////////////////////////////////////////// // // Copyright 2006 - 2015, Paul Beckingham, Federico Hernandez. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. // // http://www.opensource.org/licenses/mit-license.php // //////////////////////////////////////////////////////////////////////////////// #include #include #include #include #include #include #include #include #include #include #include #include #include #include extern Context context; static void countTasks (const std::vector &, const std::string&, int&, int&); //////////////////////////////////////////////////////////////////////////////// bool taskDiff (const Task& before, const Task& after) { // Attributes are all there is, so figure the different attribute names // between before and after. std::vector beforeAtts; for (auto& att : before) beforeAtts.push_back (att.first); std::vector afterAtts; for (auto& att : after) afterAtts.push_back (att.first); std::vector beforeOnly; std::vector afterOnly; listDiff (beforeAtts, afterAtts, beforeOnly, afterOnly); if (beforeOnly.size () != afterOnly.size ()) return true; for (auto& name : beforeAtts) if (name != "uuid" && before.get (name) != after.get (name)) return true; return false; } //////////////////////////////////////////////////////////////////////////////// std::string taskDifferences (const Task& before, const Task& after) { // Attributes are all there is, so figure the different attribute names // between before and after. std::vector beforeAtts; for (auto& att : before) beforeAtts.push_back (att.first); std::vector afterAtts; for (auto& att : after) afterAtts.push_back (att.first); std::vector beforeOnly; std::vector afterOnly; listDiff (beforeAtts, afterAtts, beforeOnly, afterOnly); // Now start generating a description of the differences. std::stringstream out; for (auto& name : beforeOnly) out << " - " << format (STRING_FEEDBACK_DELETED, ucFirst (name)) << "\n"; for (auto& name : afterOnly) { if (name == "depends") { std::vector deps_after; after.getDependencies (deps_after); std::string to; join (to, ", ", deps_after); out << " - " << format (STRING_FEEDBACK_DEP_SET, to) << "\n"; } else out << " - " << format (STRING_FEEDBACK_ATT_SET, ucFirst (name), renderAttribute (name, after.get (name))) << "\n"; } for (auto& name : beforeAtts) { // Ignore UUID differences, and find values that changed, but are not also // in the beforeOnly and afterOnly lists, which have been handled above.. if (name != "uuid" && before.get (name) != after.get (name) && std::find (beforeOnly.begin (), beforeOnly.end (), name) == beforeOnly.end () && std::find (afterOnly.begin (), afterOnly.end (), name) == afterOnly.end ()) { if (name == "depends") { std::vector deps_before; before.getDependencies (deps_before); std::string from; join (from, ", ", deps_before); std::vector deps_after; after.getDependencies (deps_after); std::string to; join (to, ", ", deps_after); out << " - " << format (STRING_FEEDBACK_DEP_MOD, from, to) << "\n"; } else out << " - " << format (STRING_FEEDBACK_ATT_MOD, ucFirst (name), renderAttribute (name, before.get (name)), renderAttribute (name, after.get (name))) << "\n"; } } // Shouldn't just say nothing. if (out.str ().length () == 0) out << " - " << STRING_FEEDBACK_NOP << "\n"; return out.str (); } //////////////////////////////////////////////////////////////////////////////// std::string taskInfoDifferences ( const Task& before, const Task& after, const std::string& dateformat, long& last_timestamp, const long current_timestamp) { // Attributes are all there is, so figure the different attribute names // between before and after. std::vector beforeAtts; for (auto& att : before) beforeAtts.push_back (att.first); std::vector afterAtts; for (auto& att : after) afterAtts.push_back (att.first); std::vector beforeOnly; std::vector afterOnly; listDiff (beforeAtts, afterAtts, beforeOnly, afterOnly); // Now start generating a description of the differences. std::stringstream out; for (auto& name : beforeOnly) { if (name == "depends") { std::vector deps_before; before.getDependencies (deps_before); std::string from; join (from, ", ", deps_before); out << format (STRING_FEEDBACK_DEP_DEL, from) << "\n"; } else if (name.substr (0, 11) == "annotation_") { out << format (STRING_FEEDBACK_ANN_DEL, before.get (name)) << "\n"; } else if (name == "start") { out << format (STRING_FEEDBACK_ATT_DEL_DUR, ucFirst (name), Duration (current_timestamp - last_timestamp).formatPrecise ()) << "\n"; } else { out << format (STRING_FEEDBACK_ATT_DEL, ucFirst (name)) << "\n"; } } for (auto& name : afterOnly) { if (name == "depends") { std::vector deps_after; after.getDependencies (deps_after); std::string to; join (to, ", ", deps_after); out << format (STRING_FEEDBACK_DEP_WAS_SET, to) << "\n"; } else if (name.substr (0, 11) == "annotation_") { out << format (STRING_FEEDBACK_ANN_ADD, after.get (name)) << "\n"; } else { if (name == "start") last_timestamp = current_timestamp; out << format (STRING_FEEDBACK_ATT_WAS_SET, ucFirst (name), renderAttribute (name, after.get (name), dateformat)) << "\n"; } } for (auto& name : beforeAtts) if (name != "uuid" && name != "modified" && before.get (name) != after.get (name) && before.get (name) != "" && after.get (name) != "") { if (name == "depends") { std::vector deps_before; before.getDependencies (deps_before); std::string from; join (from, ", ", deps_before); std::vector deps_after; after.getDependencies (deps_after); std::string to; join (to, ", ", deps_after); out << format (STRING_FEEDBACK_DEP_WAS_MOD, from, to) << "\n"; } else if (name.substr (0, 11) == "annotation_") { out << format (STRING_FEEDBACK_ANN_WAS_MOD, after.get (name)) << "\n"; } else out << format (STRING_FEEDBACK_ATT_WAS_MOD, ucFirst (name), renderAttribute (name, before.get (name), dateformat), renderAttribute (name, after.get (name), dateformat)) << "\n"; } // Shouldn't just say nothing. if (out.str ().length () == 0) out << STRING_FEEDBACK_WAS_NOP << "\n"; return out.str (); } //////////////////////////////////////////////////////////////////////////////// std::string renderAttribute (const std::string& name, const std::string& value, const std::string& format /* = "" */) { Column* col = context.columns[name]; if (col && col->type () == "date" && value != "") { Date d ((time_t)strtol (value.c_str (), NULL, 10)); if (format == "") { return d.toString (context.config.get ("dateformat")); } else { return d.toString (format); } } return value; } //////////////////////////////////////////////////////////////////////////////// // Implements: // void feedback_affected (const std::string& effect) { if (context.verbose ("affected")) std::cout << effect << "\n"; } //////////////////////////////////////////////////////////////////////////////// // Implements: // Deleted 3 tasks // // The 'effect' string should contain: // {1} Quantity void feedback_affected (const std::string& effect, int quantity) { if (context.verbose ("affected")) std::cout << format (effect, quantity) << "\n"; } //////////////////////////////////////////////////////////////////////////////// // Implements: // Deleting task 123 'This is a test' // // The 'effect' string should contain: // {1} ID // {2} Description void feedback_affected (const std::string& effect, const Task& task) { if (context.verbose ("affected")) { if (task.id) std::cout << format (effect, task.id, task.get ("description")) << "\n"; else std::cout << format (effect, task.get ("uuid"), task.get ("description")) << "\n"; } } //////////////////////////////////////////////////////////////////////////////// // Implements feedback when adding special tags to a task. void feedback_special_tags (const Task& task, const std::string& tag) { if (context.verbose ("special")) { std::string msg; std::string explanation; if (tag == "nocolor") msg = STRING_FEEDBACK_TAG_NOCOLOR; else if (tag == "nonag") msg = STRING_FEEDBACK_TAG_NONAG; else if (tag == "nocal") msg = STRING_FEEDBACK_TAG_NOCAL; else if (tag == "next") msg = STRING_FEEDBACK_TAG_NEXT; if (msg.length ()) { if (task.id) std::cout << format (msg, task.id) << "\n"; else std::cout << format (msg, task.get ("uuid")) << "\n"; } } } //////////////////////////////////////////////////////////////////////////////// // Called on completion, deletion and update. If this task is blocking another // task, then if this was the *only* blocking task, that other task is now // unblocked. Mention it. // // Implements: // Unblocked '' void feedback_unblocked (const Task& task) { if (context.verbose ("affected")) { // Get a list of tasks that depended on this task. std::vector blocked; dependencyGetBlocked (task, blocked); // Scan all the tasks that were blocked by this task for (auto& i : blocked) { std::vector blocking; dependencyGetBlocking (i, blocking); if (blocking.size () == 0) { if (i.id) std::cout << format (STRING_FEEDBACK_UNBLOCKED, i.id, i.get ("description")) << "\n"; else { std::string uuid = i.get ("uuid"); std::cout << format (STRING_FEEDBACK_UNBLOCKED, i.get ("uuid"), i.get ("description")) << "\n"; } } } } } /////////////////////////////////////////////////////////////////////////////// void feedback_backlog () { if (context.config.get ("taskd.server") != "" && context.verbose ("sync")) { std::vector lines = context.tdb2.backlog.get_lines (); for (auto& line : lines) { if ((line)[0] == '{') { context.footnote (STRING_FEEDBACK_BACKLOG); break; } } } } /////////////////////////////////////////////////////////////////////////////// std::string onProjectChange (Task& task, bool scope /* = true */) { std::stringstream msg; std::string project = task.get ("project"); if (project != "") { if (scope) msg << format (STRING_HELPER_PROJECT_CHANGE, project) << " "; // Count pending and done tasks, for this project. int count_pending = 0; int count_done = 0; std::vector all = context.tdb2.all_tasks (); countTasks (all, project, count_pending, count_done); // count_done count_pending percentage // ---------- ------------- ---------- // 0 0 0% // >0 0 100% // 0 >0 0% // >0 >0 calculated int percentage = 0; if (count_done == 0) percentage = 0; else if (count_pending == 0) percentage = 100; else percentage = (count_done * 100 / (count_done + count_pending)); msg << format (STRING_HELPER_PROJECT_COMPL, project, percentage) << " "; if (count_pending == 1 && count_done == 0) msg << format (STRING_HELPER_PROJECT_REM1, count_pending); else msg << format (STRING_HELPER_PROJECT_REM, count_pending, count_pending + count_done); } return msg.str (); } /////////////////////////////////////////////////////////////////////////////// std::string onProjectChange (Task& task1, Task& task2) { if (task1.get ("project") == task2.get ("project")) return onProjectChange (task1, false); std::string messages1 = onProjectChange (task1); std::string messages2 = onProjectChange (task2); if (messages1.length () && messages2.length ()) return messages1 + '\n' + messages2; return messages1 + messages2; } /////////////////////////////////////////////////////////////////////////////// std::string onExpiration (Task& task) { std::stringstream msg; if (context.verbose ("affected")) msg << format (STRING_FEEDBACK_EXPIRED, task.id, task.get ("description")); return msg.str (); } /////////////////////////////////////////////////////////////////////////////// static void countTasks ( const std::vector & all, const std::string& project, int& count_pending, int& count_done) { for (auto& it : all) { if (it.get ("project") == project) { switch (it.getStatus ()) { case Task::pending: case Task::waiting: ++count_pending; break; case Task::completed: ++count_done; break; case Task::deleted: case Task::recurring: default: break; } } } } ////////////////////////////////////////////////////////////////////////////////