diff --git a/convert_v3_to_v4.py b/convert_v3_to_v4.py index 0866ffcde..6aa62c48b 100755 --- a/convert_v3_to_v4.py +++ b/convert_v3_to_v4.py @@ -23,6 +23,7 @@ def strtobool(val: typing.Union[str, int, bool]) -> bool: SCRIPT_DIRECTIVES = [ "_successIf", "_failureIf", + "_onlyIf", "_skipIf", "_while", "_onSuccess", diff --git a/include/behaviortree_cpp/tree_node.h b/include/behaviortree_cpp/tree_node.h index 344b04427..94a9bb618 100644 --- a/include/behaviortree_cpp/tree_node.h +++ b/include/behaviortree_cpp/tree_node.h @@ -47,6 +47,7 @@ enum class PreCond // order of the enums also tell us the execution order FAILURE_IF = 0, SUCCESS_IF, + ONLY_IF, SKIP_IF, WHILE_TRUE, COUNT_ diff --git a/src/tree_node.cpp b/src/tree_node.cpp index 3adf9d61e..3e22f6dca 100644 --- a/src/tree_node.cpp +++ b/src/tree_node.cpp @@ -194,6 +194,7 @@ Expected TreeNode::checkPreConditions() } const PreCond preID = PreCond(index); + bool onlyIfIsPresent = false; // Some preconditions are applied only when the node state is IDLE or SKIPPED if(_p->status == NodeStatus::IDLE || _p->status == NodeStatus::SKIPPED) @@ -209,7 +210,11 @@ Expected TreeNode::checkPreConditions() { return NodeStatus::SUCCESS; } - else if(preID == PreCond::SKIP_IF) + else if(preID == PreCond::ONLY_IF) + { + onlyIfIsPresent = true; + } + else if(preID == PreCond::SKIP_IF && onlyIfIsPresent) { return NodeStatus::SKIPPED; } @@ -219,6 +224,10 @@ Expected TreeNode::checkPreConditions() { return NodeStatus::SKIPPED; } + else if(preID == PreCond::ONLY_IF) + { + return NodeStatus::SKIPPED; + } } else if(_p->status == NodeStatus::RUNNING && preID == PreCond::WHILE_TRUE) { @@ -456,6 +465,8 @@ std::string toStr(const PreCond& pre) return "_successIf"; case PreCond::FAILURE_IF: return "_failureIf"; + case PreCond::ONLY_IF: + return "_onlyIf"; case PreCond::SKIP_IF: return "_skipIf"; case PreCond::WHILE_TRUE: diff --git a/src/xml_parsing.cpp b/src/xml_parsing.cpp index adb8c8247..fba2cbb22 100644 --- a/src/xml_parsing.cpp +++ b/src/xml_parsing.cpp @@ -1296,6 +1296,7 @@ std::string writeTreeXSD(const BehaviorTreeFactory& factory) + diff --git a/tests/gtest_skipping.cpp b/tests/gtest_skipping.cpp index 0f6f86907..c213158bb 100644 --- a/tests/gtest_skipping.cpp +++ b/tests/gtest_skipping.cpp @@ -32,6 +32,33 @@ TEST(SkippingLogic, Sequence) ASSERT_EQ(counters[1], 1); } +TEST(SkippingLogic, Order) +{ + BehaviorTreeFactory factory; + std::array counters; + RegisterTestTick(factory, "Test", counters); + + const std::string xml_text = R"( + + + + +