int CmdLog::execute (std::string& output) { int rc = 0; // Apply the command line modifications to the new task. A3 modifications = context.a3.extract_modifications (); Task task; modify_task_description_replace (task, modifications); task.setStatus (Task::completed); // Recurring tasks get a special status. if (task.has ("recur")) throw std::string (STRING_CMD_LOG_NO_RECUR); if (task.has ("wait")) throw std::string (STRING_CMD_LOG_NO_WAITING); context.tdb2.add (task); if (context.verbose ("project")) context.footnote (onProjectChange (task)); context.tdb2.commit (); if (context.verbose ("affected") || context.config.getBoolean ("echo.command")) // Deprecated 2.0 output = std::string (STRING_CMD_LOG_LOGGED) + "\n"; return rc; }
int CmdAdd::execute (std::string& output) { int rc = 0; // Apply the command line modifications to the new task. Task task; modify_task_description_replace (task, context.a3.extract_modifications ()); context.tdb2.add (task); if (context.verbose ("new-id")) output = format (STRING_CMD_ADD_FEEDBACK, context.tdb2.next_id ()) + "\n"; if (context.verbose ("project")) context.footnote (onProjectChange (task)); context.tdb2.commit (); return rc; }
int CmdModify::execute (std::string& output) { int rc = 0; int count = 0; // Apply filter. std::vector <Task> filtered; filter (filtered); if (filtered.size () == 0) { context.footnote (STRING_FEEDBACK_NO_TASKS_SP); return 1; } // Apply the command line modifications to the new task. A3 modifications = context.a3.extract_modifications (); if (!modifications.size ()) throw std::string (STRING_CMD_MODIFY_NEED_TEXT); // Accumulated project change notifications. std::map <std::string, std::string> projectChanges; std::vector <Task>::iterator task; for (task = filtered.begin (); task != filtered.end (); ++task) { Task before (*task); modify_task_description_replace (*task, modifications); if (taskDiff (before, *task)) { // Perform some logical consistency checks. if (task->has ("recur") && !task->has ("due") && !before.has ("due")) throw std::string (STRING_CMD_MODIFY_NO_DUE); if (before.has ("recur") && before.has ("due") && (!task->has ("due") || task->get ("due") == "")) throw std::string (STRING_CMD_MODIFY_REM_DUE); if (before.has ("recur") && (!task->has ("recur") || task->get ("recur") == "")) throw std::string (STRING_CMD_MODIFY_REC_ALWAYS); // Delete the specified task. std::string question = format (STRING_CMD_MODIFY_CONFIRM, task->id, task->get ("description")); if (permission (*task, taskDifferences (before, *task) + question, filtered.size ())) { updateRecurrenceMask (*task); dependencyChainOnModify (before, *task); ++count; feedback_affected (STRING_CMD_MODIFY_TASK, *task); feedback_unblocked (*task); context.tdb2.modify (*task); if (context.verbose ("project")) projectChanges[task->get ("project")] = onProjectChange (before, *task); // Task potentially has siblings - modify them. if (task->has ("parent")) { std::vector <Task> siblings = context.tdb2.siblings (*task); if (siblings.size () && confirm (STRING_CMD_MODIFY_RECUR)) { std::vector <Task>::iterator sibling; for (sibling = siblings.begin (); sibling != siblings.end (); ++sibling) { Task alternate (*sibling); modify_task_description_replace (*sibling, modifications); updateRecurrenceMask (*sibling); dependencyChainOnModify (alternate, *sibling); ++count; feedback_affected (STRING_CMD_MODIFY_TASK_R, *sibling); feedback_unblocked (*sibling); context.tdb2.modify (*sibling); if (context.verbose ("project")) projectChanges[sibling->get ("project")] = onProjectChange (alternate, *sibling); } } } // Task potentially has child tasks - modify them. else if (task->get ("status") == "recurring") { std::vector <Task> children = context.tdb2.children (*task); if (children.size () && confirm (STRING_CMD_MODIFY_RECUR)) { std::vector <Task>::iterator child; for (child = children.begin (); child != children.end (); ++child) { Task alternate (*child); modify_task_description_replace (*child, modifications); updateRecurrenceMask (*child); context.tdb2.modify (*child); dependencyChainOnModify (alternate, *child); if (context.verbose ("project")) projectChanges[child->get ("project")] = onProjectChange (alternate, *child); ++count; feedback_affected (STRING_CMD_MODIFY_TASK_R, *child); } } } } else { std::cout << STRING_CMD_MODIFY_NO << "\n"; rc = 1; } } } // Now list the project changes. std::map <std::string, std::string>::iterator i; for (i = projectChanges.begin (); i != projectChanges.end (); ++i) if (i->first != "") context.footnote (i->second); context.tdb2.commit (); feedback_affected (count == 1 ? STRING_CMD_MODIFY_1 : STRING_CMD_MODIFY_N, count); return rc; }