2024-01-23 09:33:57 +00:00
|
|
|
/**
|
2024-01-23 19:02:23 +00:00
|
|
|
* blackboard_plan.cpp
|
2024-01-23 09:33:57 +00:00
|
|
|
* =============================================================================
|
|
|
|
* Copyright 2021-2024 Serhii Snitsaruk
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
* =============================================================================
|
|
|
|
*/
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
#include "blackboard_plan.h"
|
2024-01-23 09:33:57 +00:00
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
bool BlackboardPlan::_set(const StringName &p_name, const Variant &p_value) {
|
2024-01-23 14:31:56 +00:00
|
|
|
String prop_name = p_name;
|
|
|
|
|
|
|
|
// * Editor
|
2024-01-25 16:59:38 +00:00
|
|
|
if (var_map.has(prop_name)) {
|
|
|
|
var_map[prop_name].set_value(p_value);
|
2024-01-23 14:31:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// * Storage
|
|
|
|
if (prop_name.begins_with("var/")) {
|
|
|
|
String var_name = prop_name.get_slicec('/', 1);
|
|
|
|
String what = prop_name.get_slicec('/', 2);
|
2024-01-25 16:59:38 +00:00
|
|
|
if (!var_map.has(var_name) && what == "name") {
|
|
|
|
add_var(var_name, BBVariable());
|
2024-01-23 14:31:56 +00:00
|
|
|
}
|
|
|
|
if (what == "name") {
|
|
|
|
// We don't store variable name with the variable.
|
|
|
|
} else if (what == "type") {
|
2024-01-25 16:59:38 +00:00
|
|
|
var_map[var_name].set_type((Variant::Type)(int)p_value);
|
2024-01-23 14:31:56 +00:00
|
|
|
} else if (what == "value") {
|
2024-01-25 16:59:38 +00:00
|
|
|
var_map[var_name].set_value(p_value);
|
2024-01-23 14:31:56 +00:00
|
|
|
} else if (what == "hint") {
|
2024-01-25 16:59:38 +00:00
|
|
|
var_map[var_name].set_hint((PropertyHint)(int)p_value);
|
2024-01-23 14:31:56 +00:00
|
|
|
} else if (what == "hint_string") {
|
2024-01-25 16:59:38 +00:00
|
|
|
var_map[var_name].set_hint_string(p_value);
|
2024-01-23 14:31:56 +00:00
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
bool BlackboardPlan::_get(const StringName &p_name, Variant &r_ret) const {
|
2024-01-23 14:31:56 +00:00
|
|
|
String prop_name = p_name;
|
|
|
|
|
|
|
|
// * Editor
|
2024-01-25 16:59:38 +00:00
|
|
|
if (var_map.has(prop_name)) {
|
|
|
|
r_ret = var_map[prop_name].get_value();
|
2024-01-23 14:31:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// * Storage
|
|
|
|
if (!prop_name.begins_with("var/")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
String var_name = prop_name.get_slicec('/', 1);
|
|
|
|
String what = prop_name.get_slicec('/', 2);
|
2024-01-25 16:59:38 +00:00
|
|
|
ERR_FAIL_COND_V(!var_map.has(var_name), false);
|
2024-01-25 13:27:14 +00:00
|
|
|
|
|
|
|
if (what == "name") {
|
|
|
|
r_ret = var_name;
|
|
|
|
} else if (what == "type") {
|
2024-01-25 16:59:38 +00:00
|
|
|
r_ret = var_map[var_name].get_type();
|
2024-01-23 14:31:56 +00:00
|
|
|
} else if (what == "value") {
|
2024-01-25 16:59:38 +00:00
|
|
|
r_ret = var_map[var_name].get_value();
|
2024-01-23 14:31:56 +00:00
|
|
|
} else if (what == "hint") {
|
2024-01-25 16:59:38 +00:00
|
|
|
r_ret = var_map[var_name].get_hint();
|
2024-01-23 14:31:56 +00:00
|
|
|
} else if (what == "hint_string") {
|
2024-01-25 16:59:38 +00:00
|
|
|
r_ret = var_map[var_name].get_hint_string();
|
2024-01-23 14:31:56 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
void BlackboardPlan::_get_property_list(List<PropertyInfo> *p_list) const {
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : var_list) {
|
|
|
|
String var_name = p.first;
|
|
|
|
BBVariable var = p.second;
|
2024-01-23 14:31:56 +00:00
|
|
|
|
|
|
|
// * Editor
|
2024-01-25 10:51:35 +00:00
|
|
|
if (!is_derived() || !var_name.begins_with("_")) {
|
|
|
|
p_list->push_back(PropertyInfo(var.get_type(), var_name, var.get_hint(), var.get_hint_string(), PROPERTY_USAGE_EDITOR));
|
|
|
|
}
|
2024-01-23 14:31:56 +00:00
|
|
|
|
|
|
|
// * Storage
|
|
|
|
p_list->push_back(PropertyInfo(Variant::STRING, "var/" + var_name + "/name", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NO_EDITOR | PROPERTY_USAGE_INTERNAL));
|
|
|
|
p_list->push_back(PropertyInfo(Variant::INT, "var/" + var_name + "/type", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NO_EDITOR | PROPERTY_USAGE_INTERNAL));
|
|
|
|
p_list->push_back(PropertyInfo(var.get_type(), "var/" + var_name + "/value", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NO_EDITOR | PROPERTY_USAGE_INTERNAL));
|
|
|
|
p_list->push_back(PropertyInfo(Variant::INT, "var/" + var_name + "/hint", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NO_EDITOR | PROPERTY_USAGE_INTERNAL));
|
|
|
|
p_list->push_back(PropertyInfo(Variant::STRING, "var/" + var_name + "/hint_string", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NO_EDITOR | PROPERTY_USAGE_INTERNAL));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
bool BlackboardPlan::_property_can_revert(const StringName &p_name) const {
|
2024-01-25 16:59:38 +00:00
|
|
|
return base.is_valid() && base->var_map.has(p_name);
|
2024-01-23 16:54:20 +00:00
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
bool BlackboardPlan::_property_get_revert(const StringName &p_name, Variant &r_property) const {
|
2024-01-25 16:59:38 +00:00
|
|
|
if (base->var_map.has(p_name)) {
|
|
|
|
r_property = base->var_map[p_name].get_value();
|
2024-01-23 16:54:20 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
void BlackboardPlan::set_base_plan(const Ref<BlackboardPlan> &p_base) {
|
2024-01-23 14:31:56 +00:00
|
|
|
base = p_base;
|
2024-01-23 19:02:23 +00:00
|
|
|
sync_with_base_plan();
|
2024-01-23 14:31:56 +00:00
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
void BlackboardPlan::add_var(const String &p_name, const BBVariable &p_var) {
|
2024-01-25 16:59:38 +00:00
|
|
|
ERR_FAIL_COND(var_map.has(p_name));
|
|
|
|
var_map.insert(p_name, p_var);
|
|
|
|
var_list.push_back(Pair<String, BBVariable>(p_name, p_var));
|
2024-01-24 22:11:09 +00:00
|
|
|
notify_property_list_changed();
|
|
|
|
emit_changed();
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
void BlackboardPlan::remove_var(const String &p_name) {
|
2024-01-25 16:59:38 +00:00
|
|
|
ERR_FAIL_COND(!var_map.has(p_name));
|
|
|
|
var_list.erase(Pair<String, BBVariable>(p_name, var_map[p_name]));
|
|
|
|
var_map.erase(p_name);
|
2024-01-24 22:11:09 +00:00
|
|
|
notify_property_list_changed();
|
|
|
|
emit_changed();
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
BBVariable BlackboardPlan::get_var(const String &p_name) {
|
2024-01-25 16:59:38 +00:00
|
|
|
ERR_FAIL_COND_V(!var_map.has(p_name), BBVariable());
|
|
|
|
return var_map.get(p_name);
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
|
|
|
|
2024-01-24 22:11:09 +00:00
|
|
|
Pair<String, BBVariable> BlackboardPlan::get_var_by_index(int p_index) {
|
|
|
|
Pair<String, BBVariable> ret;
|
2024-01-25 16:59:38 +00:00
|
|
|
ERR_FAIL_INDEX_V(p_index, (int)var_map.size(), ret);
|
|
|
|
return var_list[p_index];
|
2024-01-24 22:11:09 +00:00
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
PackedStringArray BlackboardPlan::list_vars() const {
|
2024-01-23 09:33:57 +00:00
|
|
|
PackedStringArray ret;
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : var_list) {
|
|
|
|
ret.append(p.first);
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2024-01-24 22:11:09 +00:00
|
|
|
String BlackboardPlan::get_var_name(const BBVariable &p_var) const {
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : var_list) {
|
|
|
|
if (p.second == p_var) {
|
|
|
|
return p.first;
|
2024-01-24 22:11:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return String();
|
|
|
|
}
|
|
|
|
|
|
|
|
void BlackboardPlan::rename_var(const String &p_name, const String &p_new_name) {
|
|
|
|
ERR_FAIL_COND(p_new_name.is_empty());
|
2024-01-25 16:59:38 +00:00
|
|
|
ERR_FAIL_COND(var_map.has(p_new_name));
|
|
|
|
ERR_FAIL_COND(!var_map.has(p_name));
|
|
|
|
|
|
|
|
BBVariable var = var_map[p_name];
|
|
|
|
Pair<String, BBVariable> new_entry(p_new_name, var);
|
|
|
|
Pair<String, BBVariable> old_entry(p_name, var);
|
|
|
|
var_list.find(old_entry)->set(new_entry);
|
|
|
|
|
|
|
|
var_map.erase(p_name);
|
|
|
|
var_map.insert(p_new_name, var);
|
2024-01-24 22:11:09 +00:00
|
|
|
|
|
|
|
notify_property_list_changed();
|
|
|
|
emit_changed();
|
|
|
|
}
|
|
|
|
|
2024-01-25 21:01:14 +00:00
|
|
|
void BlackboardPlan::move_var(int p_index, int p_new_index) {
|
|
|
|
ERR_FAIL_INDEX(p_index, (int)var_map.size());
|
|
|
|
ERR_FAIL_INDEX(p_new_index, (int)var_map.size());
|
2024-01-24 22:11:09 +00:00
|
|
|
|
2024-01-25 21:01:14 +00:00
|
|
|
if (p_index == p_new_index) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
List<Pair<String, BBVariable>>::Element *E = var_list.front();
|
|
|
|
for (int i = 0; i < p_index; i++) {
|
|
|
|
E = E->next();
|
|
|
|
}
|
|
|
|
List<Pair<String, BBVariable>>::Element *E2 = var_list.front();
|
|
|
|
for (int i = 0; i < p_new_index; i++) {
|
|
|
|
E2 = E2->next();
|
|
|
|
}
|
2024-01-24 22:11:09 +00:00
|
|
|
|
2024-01-25 21:01:14 +00:00
|
|
|
var_list.move_before(E, E2);
|
|
|
|
if (p_new_index > p_index) {
|
|
|
|
var_list.move_before(E2, E);
|
|
|
|
}
|
2024-01-24 22:11:09 +00:00
|
|
|
|
|
|
|
notify_property_list_changed();
|
|
|
|
emit_changed();
|
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
void BlackboardPlan::sync_with_base_plan() {
|
2024-01-23 16:54:20 +00:00
|
|
|
if (base.is_null()) {
|
|
|
|
return;
|
|
|
|
}
|
2024-01-23 19:02:23 +00:00
|
|
|
|
2024-01-24 22:11:09 +00:00
|
|
|
bool changed = false;
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
// Sync variables with the base plan.
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : base->var_list) {
|
|
|
|
const String &base_name = p.first;
|
|
|
|
const BBVariable &base_var = p.second;
|
|
|
|
|
|
|
|
if (!var_map.has(base_name)) {
|
|
|
|
add_var(base_name, base_var.duplicate());
|
2024-01-24 22:11:09 +00:00
|
|
|
changed = true;
|
2024-01-23 09:33:57 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-25 16:59:38 +00:00
|
|
|
BBVariable var = var_map[base_name];
|
|
|
|
if (!var.is_same_prop_info(base_var)) {
|
|
|
|
var.copy_prop_info(base_var);
|
2024-01-24 22:11:09 +00:00
|
|
|
changed = true;
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
2024-01-25 16:59:38 +00:00
|
|
|
if (var.get_value().get_type() != base_var.get_type()) {
|
|
|
|
var.set_value(base_var.get_value());
|
2024-01-24 22:11:09 +00:00
|
|
|
changed = true;
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
|
|
|
}
|
2024-01-23 19:02:23 +00:00
|
|
|
|
|
|
|
// Erase variables that do not exist in the base plan.
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : var_list) {
|
|
|
|
if (!base->has_var(p.first)) {
|
|
|
|
remove_var(p.first);
|
2024-01-24 22:11:09 +00:00
|
|
|
changed = true;
|
2024-01-23 19:02:23 +00:00
|
|
|
}
|
|
|
|
}
|
2024-01-24 22:11:09 +00:00
|
|
|
|
|
|
|
if (changed) {
|
|
|
|
notify_property_list_changed();
|
|
|
|
emit_changed();
|
|
|
|
}
|
2024-01-23 09:33:57 +00:00
|
|
|
}
|
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
Ref<Blackboard> BlackboardPlan::create_blackboard() {
|
2024-01-23 09:33:57 +00:00
|
|
|
Ref<Blackboard> bb = memnew(Blackboard);
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : var_list) {
|
|
|
|
bb->add_var(p.first, p.second.duplicate());
|
2024-01-23 11:05:54 +00:00
|
|
|
}
|
2024-01-23 09:33:57 +00:00
|
|
|
return bb;
|
|
|
|
}
|
2024-01-23 11:05:54 +00:00
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
void BlackboardPlan::populate_blackboard(const Ref<Blackboard> &p_blackboard, bool overwrite) {
|
2024-01-25 16:59:38 +00:00
|
|
|
for (const Pair<String, BBVariable> &p : var_list) {
|
|
|
|
if (p_blackboard->has_var(p.first)) {
|
2024-01-23 11:05:54 +00:00
|
|
|
if (overwrite) {
|
2024-01-25 16:59:38 +00:00
|
|
|
p_blackboard->erase_var(p.first);
|
2024-01-23 11:05:54 +00:00
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2024-01-25 16:59:38 +00:00
|
|
|
p_blackboard->add_var(p.first, p.second.duplicate());
|
2024-01-23 11:05:54 +00:00
|
|
|
}
|
|
|
|
}
|
2024-01-23 14:31:56 +00:00
|
|
|
|
2024-01-23 19:02:23 +00:00
|
|
|
BlackboardPlan::BlackboardPlan() {
|
2024-01-23 14:31:56 +00:00
|
|
|
}
|