limboai/bt/tasks/utility/bt_call_method.cpp

159 lines
6.0 KiB
C++
Raw Normal View History

2023-08-15 11:09:18 +00:00
/**
* bt_call_method.cpp
* =============================================================================
* Copyright (c) 2023-present Serhii Snitsaruk and the LimboAI contributors.
2023-08-15 11:09:18 +00:00
*
* Use of this source code is governed by an MIT-style
* license that can be found in the LICENSE file or at
* https://opensource.org/licenses/MIT.
* =============================================================================
*/
#include "bt_call_method.h"
2024-01-10 21:45:42 +00:00
#include "../../../util/limbo_compat.h"
#include "../../../util/limbo_utility.h"
2024-01-10 21:45:42 +00:00
#ifdef LIMBOAI_GDEXTENSION
#include "godot_cpp/classes/global_constants.hpp"
#endif // LIMBOAI_GDEXTENSION
2023-08-15 11:09:18 +00:00
//**** Setters / Getters
void BTCallMethod::set_method(const StringName &p_method_name) {
method = p_method_name;
2023-08-15 11:09:18 +00:00
emit_changed();
}
void BTCallMethod::set_node_param(const Ref<BBNode> &p_object) {
2023-08-15 11:09:18 +00:00
node_param = p_object;
emit_changed();
if (Engine::get_singleton()->is_editor_hint() && node_param.is_valid() &&
!node_param->is_connected(LW_NAME(changed), callable_mp((Resource *)this, &Resource::emit_changed))) {
node_param->connect(LW_NAME(changed), callable_mp((Resource *)this, &Resource::emit_changed));
2023-08-15 11:09:18 +00:00
}
}
void BTCallMethod::set_include_delta(bool p_include_delta) {
include_delta = p_include_delta;
emit_changed();
}
void BTCallMethod::set_args(TypedArray<BBVariant> p_args) {
2023-08-15 11:09:18 +00:00
args = p_args;
emit_changed();
}
void BTCallMethod::set_result_var(const StringName &p_result_var) {
result_var = p_result_var;
emit_changed();
}
2023-08-15 11:09:18 +00:00
//**** Task Implementation
PackedStringArray BTCallMethod::get_configuration_warnings() {
PackedStringArray warnings = BTAction::get_configuration_warnings();
if (method == StringName()) {
warnings.append("Method Name is not set.");
2023-08-15 11:09:18 +00:00
}
if (node_param.is_null()) {
warnings.append("Node parameter is not set.");
} else if (node_param->get_value_source() == BBParam::SAVED_VALUE && node_param->get_saved_value() == Variant()) {
warnings.append("Path to node is not set.");
2023-08-15 11:09:18 +00:00
} else if (node_param->get_value_source() == BBParam::BLACKBOARD_VAR && node_param->get_variable() == StringName()) {
warnings.append("Node blackboard variable is not set.");
2023-08-15 11:09:18 +00:00
}
return warnings;
}
String BTCallMethod::_generate_name() {
String args_str = include_delta ? "delta" : "";
if (args.size() > 0) {
if (!args_str.is_empty()) {
args_str += ", ";
}
args_str += vformat("%s", args).trim_prefix("[").trim_suffix("]");
}
return vformat("CallMethod %s(%s) node: %s %s",
method != StringName() ? method : "???",
args_str,
node_param.is_valid() && !node_param->to_string().is_empty() ? node_param->to_string() : "???",
result_var == StringName() ? "" : LimboUtility::get_singleton()->decorate_output_var(result_var));
2023-08-15 11:09:18 +00:00
}
BT::Status BTCallMethod::_tick(double p_delta) {
ERR_FAIL_COND_V_MSG(method == StringName(), FAILURE, "BTCallMethod: Method Name is not set.");
2023-08-15 11:09:18 +00:00
ERR_FAIL_COND_V_MSG(node_param.is_null(), FAILURE, "BTCallMethod: Node parameter is not set.");
2024-05-01 21:32:44 +00:00
Object *obj = node_param->get_value(get_scene_root(), get_blackboard());
2023-09-05 08:55:17 +00:00
ERR_FAIL_COND_V_MSG(obj == nullptr, FAILURE, "BTCallMethod: Failed to get object: " + node_param->to_string());
2023-08-15 11:09:18 +00:00
Variant result;
Array call_args;
#ifdef LIMBOAI_MODULE
const Variant delta = include_delta ? Variant(p_delta) : Variant();
2023-08-15 11:09:18 +00:00
const Variant **argptrs = nullptr;
int argument_count = include_delta ? args.size() + 1 : args.size();
if (argument_count > 0) {
argptrs = (const Variant **)alloca(sizeof(Variant *) * argument_count);
if (include_delta) {
argptrs[0] = &delta;
}
2023-08-15 11:09:18 +00:00
for (int i = 0; i < args.size(); i++) {
Ref<BBVariant> param = args[i];
2024-05-01 21:32:44 +00:00
call_args.push_back(param->get_value(get_scene_root(), get_blackboard()));
argptrs[i + int(include_delta)] = &call_args[i];
2023-08-15 11:09:18 +00:00
}
}
Callable::CallError ce;
result = obj->callp(method, argptrs, argument_count, ce);
2023-08-15 11:09:18 +00:00
if (ce.error != Callable::CallError::CALL_OK) {
ERR_FAIL_V_MSG(FAILURE, "BTCallMethod: Error calling method: " + Variant::get_call_error_text(obj, method, argptrs, argument_count, ce) + ".");
2023-08-15 11:09:18 +00:00
}
2024-01-13 16:10:42 +00:00
#elif LIMBOAI_GDEXTENSION
if (include_delta) {
call_args.push_back(Variant(p_delta));
}
for (int i = 0; i < args.size(); i++) {
Ref<BBVariant> param = args[i];
2024-05-01 21:32:44 +00:00
call_args.push_back(param->get_value(get_scene_root(), get_blackboard()));
}
// TODO: Unsure how to detect call error, so we return SUCCESS for now...
result = obj->callv(method, call_args);
2024-01-13 16:10:42 +00:00
#endif // LIMBOAI_MODULE & LIMBOAI_GDEXTENSION
2023-08-15 11:09:18 +00:00
if (result_var != StringName()) {
get_blackboard()->set_var(result_var, result);
}
2023-08-15 11:09:18 +00:00
return SUCCESS;
}
//**** Godot
void BTCallMethod::_bind_methods() {
ClassDB::bind_method(D_METHOD("set_method", "method_name"), &BTCallMethod::set_method);
ClassDB::bind_method(D_METHOD("get_method"), &BTCallMethod::get_method);
ClassDB::bind_method(D_METHOD("set_node_param", "param"), &BTCallMethod::set_node_param);
2023-08-15 11:09:18 +00:00
ClassDB::bind_method(D_METHOD("get_node_param"), &BTCallMethod::get_node_param);
ClassDB::bind_method(D_METHOD("set_args", "args"), &BTCallMethod::set_args);
2023-08-15 11:09:18 +00:00
ClassDB::bind_method(D_METHOD("get_args"), &BTCallMethod::get_args);
ClassDB::bind_method(D_METHOD("set_include_delta", "include_delta"), &BTCallMethod::set_include_delta);
ClassDB::bind_method(D_METHOD("is_delta_included"), &BTCallMethod::is_delta_included);
ClassDB::bind_method(D_METHOD("set_result_var", "variable"), &BTCallMethod::set_result_var);
ClassDB::bind_method(D_METHOD("get_result_var"), &BTCallMethod::get_result_var);
2023-08-15 11:09:18 +00:00
ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "node", PROPERTY_HINT_RESOURCE_TYPE, "BBNode"), "set_node_param", "get_node_param");
ADD_PROPERTY(PropertyInfo(Variant::STRING_NAME, "method"), "set_method", "get_method");
ADD_PROPERTY(PropertyInfo(Variant::STRING_NAME, "result_var"), "set_result_var", "get_result_var");
ADD_GROUP("Arguments", "args_");
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "args_include_delta"), "set_include_delta", "is_delta_included");
ADD_PROPERTY(PropertyInfo(Variant::ARRAY, "args", PROPERTY_HINT_ARRAY_TYPE, RESOURCE_TYPE_HINT("BBVariant")), "set_args", "get_args");
}
BTCallMethod::BTCallMethod() {
2023-08-15 11:09:18 +00:00
}