//////////////////////////////////////////////////////////////////////////////// // 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 // //////////////////////////////////////////////////////////////////////////////// #define L10N // Localization complete. #include #include #include #include #include #include #include #include extern Context context; //////////////////////////////////////////////////////////////////////////////// CmdDelete::CmdDelete () { _keyword = "delete"; _usage = "task delete []"; _description = STRING_CMD_DELETE_USAGE; _read_only = false; _displays_id = false; } //////////////////////////////////////////////////////////////////////////////// int CmdDelete::execute (std::string& output) { int rc = 0; int count = 0; std::stringstream out; std::vector tasks; context.tdb.lock (context.config.getBoolean ("locking")); context.tdb.loadPending (tasks); // Apply filter. std::vector filtered; filter (tasks, 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 (); std::vector ::iterator task; for (task = filtered.begin (); task != filtered.end (); ++task) { if (task->getStatus () == Task::pending || task->getStatus () == Task::waiting) { modify_task_annotate (*task, modifications); apply_defaults (*task); std::string question = format (STRING_CMD_DELETE_QUESTION, task->id, task->get ("description")); if (!context.config.getBoolean ("confirmation") || confirm (question)) { // Check for the more complex case of a recurring task. If this is a // recurring task, get confirmation to delete them all. std::string parent = task->get ("parent"); if (parent != "") { if (context.config.getBoolean ("confirmation") && confirm (STRING_CMD_DELETE_CONF_RECUR)) { // Scan all pending tasks for siblings of this task, and the parent // itself, and delete them. std::vector ::iterator sibling; for (sibling = tasks.begin (); sibling != tasks.end (); ++sibling) { if (sibling->get ("parent") == parent || sibling->get ("uuid") == parent) { sibling->setStatus (Task::deleted); // Don't want a 'delete' to clobber the end date that may have // been written by a 'done' command. if (! sibling->has ("end")) sibling->setEnd (); // Apply the command line modifications to the sibling. modify_task_annotate (*sibling, modifications); apply_defaults (*sibling); sibling->validate (); context.tdb.update (*sibling); ++count; // TODO Feedback. if (context.config.getBoolean ("echo.command")) out << format (STRING_CMD_DELETE_RECURRING, sibling->id, sibling->get ("description")) << "\n"; } } } else { // Update mask in parent. task->setStatus (Task::deleted); updateRecurrenceMask (tasks, *task); // Don't want a 'delete' to clobber the end date that may have // been written by a 'done' command. if (! task->has ("end")) task->setEnd (); task->validate (); context.tdb.update (*task); ++count; out << format (STRING_CMD_DELETE_RECURRING, task->id, task->get ("description")) << "\n"; dependencyChainOnComplete (*task); context.footnote (onProjectChange (*task)); } } else { task->setStatus (Task::deleted); // Don't want a 'delete' to clobber the end date that may have // been written by a 'done' command. if (! task->has ("end")) task->setEnd (); task->validate (); context.tdb.update (*task); ++count; if (context.config.getBoolean ("echo.command")) out << format (STRING_CMD_DELETE_DELETING, task->id, task->get ("description")) << "\n"; dependencyChainOnComplete (*task); context.footnote (onProjectChange (*task)); } } else { out << STRING_CMD_DELETE_NOT << "\n"; rc = 1; } } else { out << format (STRING_CMD_DELETE_NOTPEND, task->id, task->get ("description")) << "\n"; rc = 1; } } if (count) context.tdb.commit (); context.tdb.unlock (); output = out.str (); return rc; } ////////////////////////////////////////////////////////////////////////////////