mirror of
https://github.com/GothenburgBitFactory/taskwarrior.git
synced 2025-09-03 16:17:19 +02:00

* Declare in corresponding header files and dissolve main.h Apply include-what-you-use * Remove further unncessary includes * Incorporate review comment * Do not declare static functions and variables in header * Adapt test * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
207 lines
7.7 KiB
C++
207 lines
7.7 KiB
C++
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright 2006 - 2021, Tomas Babej, 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.
|
|
//
|
|
// https://www.opensource.org/licenses/mit-license.php
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include <cmake.h>
|
|
// cmake.h include header must come first
|
|
|
|
#include <CmdModify.h>
|
|
#include <Context.h>
|
|
#include <Filter.h>
|
|
#include <feedback.h>
|
|
#include <format.h>
|
|
#include <recur.h>
|
|
#include <shared.h>
|
|
|
|
#include <iostream>
|
|
|
|
#define STRING_CMD_MODIFY_TASK_R "Modifying recurring task {1} '{2}'."
|
|
#define STRING_CMD_MODIFY_RECUR \
|
|
"This is a recurring task. Do you want to modify all pending recurrences of this same task?"
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
CmdModify::CmdModify() {
|
|
_keyword = "modify";
|
|
_usage = "task <filter> modify <mods>";
|
|
_description = "Modifies the existing task with provided arguments.";
|
|
_read_only = false;
|
|
_displays_id = false;
|
|
_needs_gc = false;
|
|
_needs_recur_update = false;
|
|
_uses_context = false;
|
|
_accepts_filter = true;
|
|
_accepts_modifications = true;
|
|
_accepts_miscellaneous = false;
|
|
_category = Command::Category::operation;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
int CmdModify::execute(std::string &) {
|
|
auto rc = 0;
|
|
|
|
// Apply filter.
|
|
Filter filter;
|
|
std::vector<Task> filtered;
|
|
filter.subset(filtered);
|
|
if (filtered.size() == 0) {
|
|
Context::getContext().footnote("No tasks specified.");
|
|
return 1;
|
|
}
|
|
|
|
// Accumulated project change notifications.
|
|
std::map<std::string, std::string> projectChanges;
|
|
|
|
auto count = 0;
|
|
if (filtered.size() > 1) {
|
|
feedback_affected("This command will alter {1} tasks.", filtered.size());
|
|
}
|
|
for (auto &task : filtered) {
|
|
Task before(task);
|
|
task.modify(Task::modReplace);
|
|
|
|
if (before != task) {
|
|
// Abort if change introduces inconsistencies.
|
|
checkConsistency(before, task);
|
|
|
|
auto question =
|
|
format("Modify task {1} '{2}'?", task.identifier(true), task.get("description"));
|
|
|
|
if (permission(before.diff(task) + question, filtered.size())) {
|
|
count += modifyAndUpdate(before, task, &projectChanges);
|
|
} else {
|
|
std::cout << "Task not modified.\n";
|
|
rc = 1;
|
|
if (_permission_quit) break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now list the project changes.
|
|
for (const auto &change : projectChanges)
|
|
if (change.first != "") Context::getContext().footnote(change.second);
|
|
|
|
feedback_affected(count == 1 ? "Modified {1} task." : "Modified {1} tasks.", count);
|
|
return rc;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// TODO Why is this not in Task::validate?
|
|
void CmdModify::checkConsistency(Task &before, Task &after) {
|
|
// Perform some logical consistency checks.
|
|
if (after.has("recur") && !after.has("due") && !before.has("due"))
|
|
throw std::string("You cannot specify a recurring task without a due date.");
|
|
|
|
if (before.has("recur") && before.has("due") && (!after.has("due") || after.get("due") == ""))
|
|
throw std::string("You cannot remove the due date from a recurring task.");
|
|
|
|
if (before.has("recur") && (!after.has("recur") || after.get("recur") == ""))
|
|
throw std::string("You cannot remove the recurrence from a recurring task.");
|
|
|
|
if ((before.getStatus() == Task::pending) && (after.getStatus() == Task::pending) &&
|
|
(before.get("end") == "") && (after.get("end") != ""))
|
|
throw format("Could not modify task {1}. You cannot set an end date on a pending task.",
|
|
before.identifier(true));
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
int CmdModify::modifyAndUpdate(Task &before, Task &after,
|
|
std::map<std::string, std::string> *projectChanges /* = NULL */) {
|
|
// This task.
|
|
auto count = 1;
|
|
|
|
updateRecurrenceMask(after);
|
|
feedback_affected("Modifying task {1} '{2}'.", after);
|
|
feedback_unblocked(after);
|
|
Context::getContext().tdb2.modify(after);
|
|
if (Context::getContext().verbose("project") && projectChanges)
|
|
(*projectChanges)[after.get("project")] = onProjectChange(before, after);
|
|
|
|
// Task has siblings - modify them.
|
|
if (after.has("parent")) count += modifyRecurrenceSiblings(after, projectChanges);
|
|
|
|
// Task has child tasks - modify them.
|
|
else if (after.get("status") == "recurring")
|
|
count += modifyRecurrenceParent(after, projectChanges);
|
|
|
|
return count;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
int CmdModify::modifyRecurrenceSiblings(
|
|
Task &task, std::map<std::string, std::string> *projectChanges /* = NULL */) {
|
|
auto count = 0;
|
|
|
|
if ((Context::getContext().config.get("recurrence.confirmation") == "prompt" &&
|
|
confirm(STRING_CMD_MODIFY_RECUR)) ||
|
|
Context::getContext().config.getBoolean("recurrence.confirmation")) {
|
|
std::vector<Task> siblings = Context::getContext().tdb2.siblings(task);
|
|
for (auto &sibling : siblings) {
|
|
Task alternate(sibling);
|
|
sibling.modify(Task::modReplace);
|
|
updateRecurrenceMask(sibling);
|
|
++count;
|
|
feedback_affected(STRING_CMD_MODIFY_TASK_R, sibling);
|
|
feedback_unblocked(sibling);
|
|
Context::getContext().tdb2.modify(sibling);
|
|
if (Context::getContext().verbose("project") && projectChanges)
|
|
(*projectChanges)[sibling.get("project")] = onProjectChange(alternate, sibling);
|
|
}
|
|
|
|
// Modify the parent
|
|
Task parent;
|
|
Context::getContext().tdb2.get(task.get("parent"), parent);
|
|
parent.modify(Task::modReplace);
|
|
Context::getContext().tdb2.modify(parent);
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
int CmdModify::modifyRecurrenceParent(
|
|
Task &task, std::map<std::string, std::string> *projectChanges /* = NULL */) {
|
|
auto count = 0;
|
|
|
|
auto children = Context::getContext().tdb2.children(task);
|
|
if (children.size() &&
|
|
((Context::getContext().config.get("recurrence.confirmation") == "prompt" &&
|
|
confirm(STRING_CMD_MODIFY_RECUR)) ||
|
|
Context::getContext().config.getBoolean("recurrence.confirmation"))) {
|
|
for (auto &child : children) {
|
|
Task alternate(child);
|
|
child.modify(Task::modReplace);
|
|
updateRecurrenceMask(child);
|
|
Context::getContext().tdb2.modify(child);
|
|
if (Context::getContext().verbose("project") && projectChanges)
|
|
(*projectChanges)[child.get("project")] = onProjectChange(alternate, child);
|
|
++count;
|
|
feedback_affected(STRING_CMD_MODIFY_TASK_R, child);
|
|
}
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|