/*************************************************************************/ /* visual_script_flow_control.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* http://www.godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2017 Juan Linietsky, Ariel Manzur. */ /* Copyright (c) 2014-2017 Godot Engine contributors (cf. AUTHORS.md) */ /* */ /* 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. */ /*************************************************************************/ #include "visual_script_flow_control.h" #include "project_settings.h" #include "io/resource_loader.h" #include "os/keyboard.h" ////////////////////////////////////////// ////////////////RETURN//////////////////// ////////////////////////////////////////// int VisualScriptReturn::get_output_sequence_port_count() const { return 0; } bool VisualScriptReturn::has_input_sequence_port() const { return true; } int VisualScriptReturn::get_input_value_port_count() const { return with_value ? 1 : 0; } int VisualScriptReturn::get_output_value_port_count() const { return 0; } String VisualScriptReturn::get_output_sequence_port_text(int p_port) const { return String(); } PropertyInfo VisualScriptReturn::get_input_value_port_info(int p_idx) const { PropertyInfo pinfo; pinfo.name = "result"; pinfo.type = type; return pinfo; } PropertyInfo VisualScriptReturn::get_output_value_port_info(int p_idx) const { return PropertyInfo(); } String VisualScriptReturn::get_caption() const { return "Return"; } String VisualScriptReturn::get_text() const { return get_name(); } void VisualScriptReturn::set_return_type(Variant::Type p_type) { if (type == p_type) return; type = p_type; ports_changed_notify(); } Variant::Type VisualScriptReturn::get_return_type() const { return type; } void VisualScriptReturn::set_enable_return_value(bool p_enable) { if (with_value == p_enable) return; with_value = p_enable; ports_changed_notify(); } bool VisualScriptReturn::is_return_value_enabled() const { return with_value; } void VisualScriptReturn::_bind_methods() { ClassDB::bind_method(D_METHOD("set_return_type", "type"), &VisualScriptReturn::set_return_type); ClassDB::bind_method(D_METHOD("get_return_type"), &VisualScriptReturn::get_return_type); ClassDB::bind_method(D_METHOD("set_enable_return_value", "enable"), &VisualScriptReturn::set_enable_return_value); ClassDB::bind_method(D_METHOD("is_return_value_enabled"), &VisualScriptReturn::is_return_value_enabled); String argt = "Any"; for (int i = 1; i < Variant::VARIANT_MAX; i++) { argt += "," + Variant::get_type_name(Variant::Type(i)); } ADD_PROPERTY(PropertyInfo(Variant::BOOL, "return_enabled"), "set_enable_return_value", "is_return_value_enabled"); ADD_PROPERTY(PropertyInfo(Variant::INT, "return_type", PROPERTY_HINT_ENUM, argt), "set_return_type", "get_return_type"); } class VisualScriptNodeInstanceReturn : public VisualScriptNodeInstance { public: VisualScriptReturn *node; VisualScriptInstance *instance; bool with_value; virtual int get_working_memory_size() const { return 1; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return true; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { if (with_value) { *p_working_mem = *p_inputs[0]; } else { *p_working_mem = Variant(); } return 0; } }; VisualScriptNodeInstance *VisualScriptReturn::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceReturn *instance = memnew(VisualScriptNodeInstanceReturn); instance->node = this; instance->instance = p_instance; instance->with_value = with_value; return instance; } VisualScriptReturn::VisualScriptReturn() { with_value = false; type = Variant::NIL; } template static Ref create_return_node(const String &p_name) { Ref node; node.instance(); node->set_enable_return_value(with_value); return node; } ////////////////////////////////////////// ////////////////CONDITION///////////////// ////////////////////////////////////////// int VisualScriptCondition::get_output_sequence_port_count() const { return 3; } bool VisualScriptCondition::has_input_sequence_port() const { return true; } int VisualScriptCondition::get_input_value_port_count() const { return 1; } int VisualScriptCondition::get_output_value_port_count() const { return 0; } String VisualScriptCondition::get_output_sequence_port_text(int p_port) const { if (p_port == 0) return "true"; else if (p_port == 1) return "false"; else return "done"; } PropertyInfo VisualScriptCondition::get_input_value_port_info(int p_idx) const { PropertyInfo pinfo; pinfo.name = "cond"; pinfo.type = Variant::BOOL; return pinfo; } PropertyInfo VisualScriptCondition::get_output_value_port_info(int p_idx) const { return PropertyInfo(); } String VisualScriptCondition::get_caption() const { return "Condition"; } String VisualScriptCondition::get_text() const { return "if (cond) is: "; } void VisualScriptCondition::_bind_methods() { } class VisualScriptNodeInstanceCondition : public VisualScriptNodeInstance { public: VisualScriptCondition *node; VisualScriptInstance *instance; //virtual int get_working_memory_size() const { return 1; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return true; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { if (p_start_mode == START_MODE_CONTINUE_SEQUENCE) return 2; else if (p_inputs[0]->operator bool()) return 0 | STEP_FLAG_PUSH_STACK_BIT; else return 1 | STEP_FLAG_PUSH_STACK_BIT; } }; VisualScriptNodeInstance *VisualScriptCondition::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceCondition *instance = memnew(VisualScriptNodeInstanceCondition); instance->node = this; instance->instance = p_instance; return instance; } VisualScriptCondition::VisualScriptCondition() { } ////////////////////////////////////////// ////////////////WHILE///////////////// ////////////////////////////////////////// int VisualScriptWhile::get_output_sequence_port_count() const { return 2; } bool VisualScriptWhile::has_input_sequence_port() const { return true; } int VisualScriptWhile::get_input_value_port_count() const { return 1; } int VisualScriptWhile::get_output_value_port_count() const { return 0; } String VisualScriptWhile::get_output_sequence_port_text(int p_port) const { if (p_port == 0) return "repeat"; else return "exit"; } PropertyInfo VisualScriptWhile::get_input_value_port_info(int p_idx) const { PropertyInfo pinfo; pinfo.name = "cond"; pinfo.type = Variant::BOOL; return pinfo; } PropertyInfo VisualScriptWhile::get_output_value_port_info(int p_idx) const { return PropertyInfo(); } String VisualScriptWhile::get_caption() const { return "While"; } String VisualScriptWhile::get_text() const { return "while (cond): "; } void VisualScriptWhile::_bind_methods() { } class VisualScriptNodeInstanceWhile : public VisualScriptNodeInstance { public: VisualScriptWhile *node; VisualScriptInstance *instance; //virtual int get_working_memory_size() const { return 1; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return true; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { bool keep_going = p_inputs[0]->operator bool(); if (keep_going) return 0 | STEP_FLAG_PUSH_STACK_BIT; else return 1; } }; VisualScriptNodeInstance *VisualScriptWhile::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceWhile *instance = memnew(VisualScriptNodeInstanceWhile); instance->node = this; instance->instance = p_instance; return instance; } VisualScriptWhile::VisualScriptWhile() { } ////////////////////////////////////////// ////////////////ITERATOR///////////////// ////////////////////////////////////////// int VisualScriptIterator::get_output_sequence_port_count() const { return 2; } bool VisualScriptIterator::has_input_sequence_port() const { return true; } int VisualScriptIterator::get_input_value_port_count() const { return 1; } int VisualScriptIterator::get_output_value_port_count() const { return 1; } String VisualScriptIterator::get_output_sequence_port_text(int p_port) const { if (p_port == 0) return "each"; else return "exit"; } PropertyInfo VisualScriptIterator::get_input_value_port_info(int p_idx) const { PropertyInfo pinfo; pinfo.name = "input"; pinfo.type = Variant::NIL; return pinfo; } PropertyInfo VisualScriptIterator::get_output_value_port_info(int p_idx) const { PropertyInfo pinfo; pinfo.name = "elem"; pinfo.type = Variant::NIL; return pinfo; } String VisualScriptIterator::get_caption() const { return "Iterator"; } String VisualScriptIterator::get_text() const { return "for (elem) in (input): "; } void VisualScriptIterator::_bind_methods() { } class VisualScriptNodeInstanceIterator : public VisualScriptNodeInstance { public: VisualScriptIterator *node; VisualScriptInstance *instance; virtual int get_working_memory_size() const { return 2; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return true; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { if (p_start_mode == START_MODE_BEGIN_SEQUENCE) { p_working_mem[0] = *p_inputs[0]; bool valid; bool can_iter = p_inputs[0]->iter_init(p_working_mem[1], valid); if (!valid) { r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD; r_error_str = RTR("Input type not iterable: ") + Variant::get_type_name(p_inputs[0]->get_type()); return 0; } if (!can_iter) return 1; //nothing to iterate *p_outputs[0] = p_working_mem[0].iter_get(p_working_mem[1], valid); if (!valid) { r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD; r_error_str = RTR("Iterator became invalid"); return 0; } } else { //continue sequence bool valid; bool can_iter = p_working_mem[0].iter_next(p_working_mem[1], valid); if (!valid) { r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD; r_error_str = RTR("Iterator became invalid: ") + Variant::get_type_name(p_inputs[0]->get_type()); return 0; } if (!can_iter) return 1; //nothing to iterate *p_outputs[0] = p_working_mem[0].iter_get(p_working_mem[1], valid); if (!valid) { r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD; r_error_str = RTR("Iterator became invalid"); return 0; } } return 0 | STEP_FLAG_PUSH_STACK_BIT; //go around } }; VisualScriptNodeInstance *VisualScriptIterator::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceIterator *instance = memnew(VisualScriptNodeInstanceIterator); instance->node = this; instance->instance = p_instance; return instance; } VisualScriptIterator::VisualScriptIterator() { } ////////////////////////////////////////// ////////////////SEQUENCE///////////////// ////////////////////////////////////////// int VisualScriptSequence::get_output_sequence_port_count() const { return steps; } bool VisualScriptSequence::has_input_sequence_port() const { return true; } int VisualScriptSequence::get_input_value_port_count() const { return 0; } int VisualScriptSequence::get_output_value_port_count() const { return 1; } String VisualScriptSequence::get_output_sequence_port_text(int p_port) const { return itos(p_port + 1); } PropertyInfo VisualScriptSequence::get_input_value_port_info(int p_idx) const { return PropertyInfo(); } PropertyInfo VisualScriptSequence::get_output_value_port_info(int p_idx) const { return PropertyInfo(Variant::INT, "current"); } String VisualScriptSequence::get_caption() const { return "Sequence"; } String VisualScriptSequence::get_text() const { return "in order: "; } void VisualScriptSequence::set_steps(int p_steps) { ERR_FAIL_COND(p_steps < 1); if (steps == p_steps) return; steps = p_steps; ports_changed_notify(); } int VisualScriptSequence::get_steps() const { return steps; } void VisualScriptSequence::_bind_methods() { ClassDB::bind_method(D_METHOD("set_steps", "steps"), &VisualScriptSequence::set_steps); ClassDB::bind_method(D_METHOD("get_steps"), &VisualScriptSequence::get_steps); ADD_PROPERTY(PropertyInfo(Variant::INT, "steps", PROPERTY_HINT_RANGE, "1,64,1"), "set_steps", "get_steps"); } class VisualScriptNodeInstanceSequence : public VisualScriptNodeInstance { public: VisualScriptSequence *node; VisualScriptInstance *instance; int steps; virtual int get_working_memory_size() const { return 1; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return true; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { if (p_start_mode == START_MODE_BEGIN_SEQUENCE) { p_working_mem[0] = 0; } int step = p_working_mem[0]; *p_outputs[0] = step; if (step + 1 == steps) return step; else { p_working_mem[0] = step + 1; return step | STEP_FLAG_PUSH_STACK_BIT; } } }; VisualScriptNodeInstance *VisualScriptSequence::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceSequence *instance = memnew(VisualScriptNodeInstanceSequence); instance->node = this; instance->instance = p_instance; instance->steps = steps; return instance; } VisualScriptSequence::VisualScriptSequence() { steps = 1; } ////////////////////////////////////////// ////////////////EVENT TYPE FILTER/////////// ////////////////////////////////////////// int VisualScriptSwitch::get_output_sequence_port_count() const { return case_values.size() + 1; } bool VisualScriptSwitch::has_input_sequence_port() const { return true; } int VisualScriptSwitch::get_input_value_port_count() const { return case_values.size() + 1; } int VisualScriptSwitch::get_output_value_port_count() const { return 0; } String VisualScriptSwitch::get_output_sequence_port_text(int p_port) const { if (p_port == case_values.size()) return "done"; return String(); } PropertyInfo VisualScriptSwitch::get_input_value_port_info(int p_idx) const { if (p_idx < case_values.size()) { return PropertyInfo(case_values[p_idx].type, " ="); } else return PropertyInfo(Variant::NIL, "input"); } PropertyInfo VisualScriptSwitch::get_output_value_port_info(int p_idx) const { return PropertyInfo(); } String VisualScriptSwitch::get_caption() const { return "Switch"; } String VisualScriptSwitch::get_text() const { return "'input' is:"; } class VisualScriptNodeInstanceSwitch : public VisualScriptNodeInstance { public: VisualScriptInstance *instance; int case_count; //virtual int get_working_memory_size() const { return 0; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return false; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { if (p_start_mode == START_MODE_CONTINUE_SEQUENCE) { return case_count; //exit } for (int i = 0; i < case_count; i++) { if (*p_inputs[i] == *p_inputs[case_count]) { return i | STEP_FLAG_PUSH_STACK_BIT; } } return case_count; } }; VisualScriptNodeInstance *VisualScriptSwitch::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceSwitch *instance = memnew(VisualScriptNodeInstanceSwitch); instance->instance = p_instance; instance->case_count = case_values.size(); return instance; } bool VisualScriptSwitch::_set(const StringName &p_name, const Variant &p_value) { if (String(p_name) == "case_count") { case_values.resize(p_value); _change_notify(); ports_changed_notify(); return true; } if (String(p_name).begins_with("case/")) { int idx = String(p_name).get_slice("/", 1).to_int(); ERR_FAIL_INDEX_V(idx, case_values.size(), false); case_values[idx].type = Variant::Type(int(p_value)); _change_notify(); ports_changed_notify(); return true; } return false; } bool VisualScriptSwitch::_get(const StringName &p_name, Variant &r_ret) const { if (String(p_name) == "case_count") { r_ret = case_values.size(); return true; } if (String(p_name).begins_with("case/")) { int idx = String(p_name).get_slice("/", 1).to_int(); ERR_FAIL_INDEX_V(idx, case_values.size(), false); r_ret = case_values[idx].type; return true; } return false; } void VisualScriptSwitch::_get_property_list(List *p_list) const { p_list->push_back(PropertyInfo(Variant::INT, "case_count", PROPERTY_HINT_RANGE, "0,128")); String argt = "Any"; for (int i = 1; i < Variant::VARIANT_MAX; i++) { argt += "," + Variant::get_type_name(Variant::Type(i)); } for (int i = 0; i < case_values.size(); i++) { p_list->push_back(PropertyInfo(Variant::INT, "case/" + itos(i), PROPERTY_HINT_ENUM, argt)); } } void VisualScriptSwitch::_bind_methods() { } VisualScriptSwitch::VisualScriptSwitch() { } ////////////////////////////////////////// ////////////////EVENT ACTION FILTER/////////// ////////////////////////////////////////// #if 0 int VisualScriptInputFilter::get_output_sequence_port_count() const { return filters.size(); } bool VisualScriptInputFilter::has_input_sequence_port() const { return true; } int VisualScriptInputFilter::get_input_value_port_count() const { return 1; } int VisualScriptInputFilter::get_output_value_port_count() const { return 1; } String VisualScriptInputFilter::get_output_sequence_port_text(int p_port) const { String text; switch (filters[p_port].type) { case Ref::NONE: { text = "None"; } break; case Ref::KEY: { InputEventKey k = filters[p_port].key; if (k->get_scancode() == 0 && k.unicode == 0) { text = "No Key"; } else { if (k->get_scancode() != 0) { text = "KeyCode: " + keycode_get_string(k->get_scancode()); } else if (k.unicode != 0) { text = "Uniode: " + String::chr(k.unicode); } if (k->is_pressed()) text += ", Pressed"; else text += ", Released"; if (k.echo) text += ", Echo"; if (k->get_alt()) text = "Alt+" + text; if (k->get_shift()) text = "Shift+" + text; if (k->get_control()) text = "Ctrl+" + text; if (k->get_metakey()) text = "Meta+" + text; } } break; case Ref::MOUSE_MOTION: { InputEventMouseMotion mm = filters[p_port].mouse_motion; text = "Mouse Motion"; String b = "Left,Right,Middle,WheelUp,WheelDown,WheelLeft,WheelRight"; for (int i = 0; i < 7; i++) { if (mm->get_button_mask() & (1 << i)) { text = b.get_slice(",", i) + "+" + text; } } if (mm->get_alt()) text = "Alt+" + text; if (mm->get_shift()) text = "Shift+" + text; if (mm->get_control()) text = "Ctrl+" + text; if (mm->get_metakey()) text = "Meta+" + text; } break; case Ref::MOUSE_BUTTON: { InputEventMouseButton mb = filters[p_port].mouse_button; String b = "Any,Left,Right,Middle,WheelUp,WheelDown,WheelLeft,WheelRight"; text = b.get_slice(",", mb->get_button_index()) + " Mouse Button"; if (mb->is_pressed()) text += ", Pressed"; else text += ", Released"; if (mb.doubleclick) text += ", DblClick"; if (mb->get_alt()) text = "Alt+" + text; if (mb->get_shift()) text = "Shift+" + text; if (mb->get_control()) text = "Ctrl+" + text; if (mb->get_metakey()) text = "Meta+" + text; } break; case Ref::JOYPAD_MOTION: { InputEventJoypadMotion jm = filters[p_port].joy_motion; text = "JoyMotion Axis " + itos(jm.axis >> 1); if (jm.axis & 1) text += " > " + rtos(jm.axis_value); else text += " < " + rtos(-jm.axis_value); } break; case Ref::JOYPAD_BUTTON: { InputEventJoypadButton jb = filters[p_port].joy_button; text = "JoyButton " + itos(jb->get_button_index()); if (jb->is_pressed()) text += ", Pressed"; else text += ", Released"; } break; case Ref::SCREEN_TOUCH: { InputEventScreenTouch sd = filters[p_port].screen_touch; text = "Touch Finger " + itos(sd.index); if (sd->is_pressed()) text += ", Pressed"; else text += ", Released"; } break; case Ref::SCREEN_DRAG: { InputEventScreenDrag sd = filters[p_port].screen_drag; text = "Drag Finger " + itos(sd.index); } break; case Ref::ACTION: { List pinfo; ProjectSettings::get_singleton()->get_property_list(&pinfo); int index = 1; text = "No Action"; for (List::Element *E = pinfo.front(); E; E = E->next()) { const PropertyInfo &pi = E->get(); if (!pi.name.begins_with("input/")) continue; if (filters[p_port].action.action == index) { text = "Action " + pi.name.substr(pi.name.find("/") + 1, pi.name.length()); break; } index++; } if (filters[p_port].action->is_pressed()) text += ", Pressed"; else text += ", Released"; } break; } return text + " - " + itos(p_port); } PropertyInfo VisualScriptInputFilter::get_input_value_port_info(int p_idx) const { return PropertyInfo(Variant::INPUT_EVENT, "event"); } PropertyInfo VisualScriptInputFilter::get_output_value_port_info(int p_idx) const { return PropertyInfo(Variant::INPUT_EVENT, ""); } String VisualScriptInputFilter::get_caption() const { return "InputFilter"; } String VisualScriptInputFilter::get_text() const { return ""; } bool VisualScriptInputFilter::_set(const StringName &p_name, const Variant &p_value) { if (p_name == "filter_count") { filters.resize(p_value); _change_notify(); ports_changed_notify(); return true; } if (String(p_name).begins_with("filter_")) { int idx = String(p_name).replace_first("filters_", "").get_slice("/", 0).to_int(); ERR_FAIL_INDEX_V(idx, filters.size(), false); String what = String(p_name).get_slice("/", 1); if (what == "type") { filters[idx] = Ref(); filters[idx].type = Ref::Type(int(p_value)); if (filters[idx].type == Ref::JOYPAD_MOTION) { filters[idx].joy_motion.axis_value = 0.5; //for threshold } else if (filters[idx].type == Ref::KEY) { filters[idx]->is_pressed() = true; //put these as true to make it more user friendly } else if (filters[idx].type == Ref::MOUSE_BUTTON) { filters[idx]->is_pressed() = true; } else if (filters[idx].type == Ref::JOYPAD_BUTTON) { filters[idx].joy_button->is_pressed() = true; } else if (filters[idx].type == Ref::SCREEN_TOUCH) { filters[idx].screen_touch->is_pressed() = true; } else if (filters[idx].type == Ref::ACTION) { filters[idx].action->is_pressed() = true; } _change_notify(); ports_changed_notify(); return true; } if (what == "device") { filters[idx].device = p_value; ports_changed_notify(); return true; } switch (filters[idx].type) { case Ref::KEY: { if (what == "scancode") { String sc = p_value; if (sc == String()) { filters[idx]->get_scancode() = 0; } else { filters[idx]->get_scancode() = find_keycode(p_value); } } else if (what == "unicode") { String uc = p_value; if (uc == String()) { filters[idx].key.unicode = 0; } else { filters[idx].key.unicode = uc[0]; } } else if (what == "pressed") { filters[idx]->is_pressed() = p_value; } else if (what == "echo") { filters[idx]->is_echo() = p_value; } else if (what == "mod_alt") { filters[idx]->get_alt() = p_value; } else if (what == "mod_shift") { filters[idx]->get_shift() = p_value; } else if (what == "mod_ctrl") { filters[idx]->get_control() = p_value; } else if (what == "mod_meta") { filters[idx]->get_metakey() = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::MOUSE_MOTION: { if (what == "button_mask") { filters[idx]->get_button_mask() = p_value; } else if (what == "mod_alt") { filters[idx].mouse_motion->get_alt() = p_value; } else if (what == "mod_shift") { filters[idx].mouse_motion->get_shift() = p_value; } else if (what == "mod_ctrl") { filters[idx].mouse_motion->get_control() = p_value; } else if (what == "mod_meta") { filters[idx].mouse_motion->get_metakey() = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::MOUSE_BUTTON: { if (what == "button_index") { filters[idx]->get_button_index() = p_value; } else if (what == "pressed") { filters[idx]->is_pressed() = p_value; } else if (what == "doubleclicked") { filters[idx].mouse_button.doubleclick = p_value; } else if (what == "mod_alt") { filters[idx].mouse_button->get_alt() = p_value; } else if (what == "mod_shift") { filters[idx].mouse_button->get_shift() = p_value; } else if (what == "mod_ctrl") { filters[idx].mouse_button->get_control() = p_value; } else if (what == "mod_meta") { filters[idx].mouse_button->get_metakey() = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::JOYPAD_MOTION: { if (what == "axis") { filters[idx].joy_motion.axis = int(p_value) << 1 | filters[idx].joy_motion.axis; } else if (what == "mode") { filters[idx].joy_motion.axis |= int(p_value); } else if (what == "threshold") { filters[idx].joy_motion.axis_value = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::JOYPAD_BUTTON: { if (what == "button_index") { filters[idx].joy_button->get_button_index() = p_value; } else if (what == "pressed") { filters[idx].joy_button->is_pressed() = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::SCREEN_TOUCH: { if (what == "finger_index") { filters[idx].screen_touch.index = p_value; } else if (what == "pressed") { filters[idx].screen_touch->is_pressed() = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::SCREEN_DRAG: { if (what == "finger_index") { filters[idx].screen_drag.index = p_value; } else { return false; } ports_changed_notify(); return true; } break; case Ref::ACTION: { if (what == "action_name") { List pinfo; ProjectSettings::get_singleton()->get_property_list(&pinfo); int index = 1; for (List::Element *E = pinfo.front(); E; E = E->next()) { const PropertyInfo &pi = E->get(); if (!pi.name.begins_with("input/")) continue; String name = pi.name.substr(pi.name.find("/") + 1, pi.name.length()); if (name == String(p_value)) { filters[idx].action.action = index; ports_changed_notify(); return true; } index++; } filters[idx].action.action = 0; ports_changed_notify(); return false; } else if (what == "pressed") { filters[idx].action->is_pressed() = p_value; ports_changed_notify(); return true; } } break; } } return false; } bool VisualScriptInputFilter::_get(const StringName &p_name, Variant &r_ret) const { if (p_name == "filter_count") { r_ret = filters.size(); return true; } if (String(p_name).begins_with("filter_")) { int idx = String(p_name).replace_first("filters_", "").get_slice("/", 0).to_int(); ERR_FAIL_INDEX_V(idx, filters.size(), false); String what = String(p_name).get_slice("/", 1); if (what == "type") { r_ret = filters[idx].type; return true; } if (what == "device") { r_ret = filters[idx].device; return true; } switch (filters[idx].type) { case Ref::KEY: { if (what == "scancode") { if (filters[idx]->get_scancode() == 0) r_ret = String(); else { r_ret = keycode_get_string(filters[idx]->get_scancode()); } } else if (what == "unicode") { if (filters[idx].key.unicode == 0) { r_ret = String(); } else { CharType str[2] = { (CharType)filters[idx].key.unicode, 0 }; r_ret = String(str); } } else if (what == "pressed") { r_ret = filters[idx]->is_pressed(); } else if (what == "echo") { r_ret = filters[idx]->is_echo(); } else if (what == "mod_alt") { r_ret = filters[idx]->get_alt(); } else if (what == "mod_shift") { r_ret = filters[idx]->get_shift(); } else if (what == "mod_ctrl") { r_ret = filters[idx]->get_control(); } else if (what == "mod_meta") { r_ret = filters[idx]->get_metakey(); } else { return false; } return true; } break; case Ref::MOUSE_MOTION: { if (what == "button_mask") { r_ret = filters[idx]->get_button_mask(); } else if (what == "mod_alt") { r_ret = filters[idx].mouse_motion->get_alt(); } else if (what == "mod_shift") { r_ret = filters[idx].mouse_motion->get_shift(); } else if (what == "mod_ctrl") { r_ret = filters[idx].mouse_motion->get_control(); } else if (what == "mod_meta") { r_ret = filters[idx].mouse_motion->get_metakey(); } else { return false; } return true; } break; case Ref::MOUSE_BUTTON: { if (what == "button_index") { r_ret = filters[idx]->get_button_index(); } else if (what == "pressed") { r_ret = filters[idx]->is_pressed(); } else if (what == "doubleclicked") { r_ret = filters[idx].mouse_button.doubleclick; } else if (what == "mod_alt") { r_ret = filters[idx].mouse_button->get_alt(); } else if (what == "mod_shift") { r_ret = filters[idx].mouse_button->get_shift(); } else if (what == "mod_ctrl") { r_ret = filters[idx].mouse_button->get_control(); } else if (what == "mod_meta") { r_ret = filters[idx].mouse_button->get_metakey(); } else { return false; } return true; } break; case Ref::JOYPAD_MOTION: { if (what == "axis_index") { r_ret = filters[idx].joy_motion.axis >> 1; } else if (what == "mode") { r_ret = filters[idx].joy_motion.axis & 1; } else if (what == "threshold") { r_ret = filters[idx].joy_motion.axis_value; } else { return false; } return true; } break; case Ref::JOYPAD_BUTTON: { if (what == "button_index") { r_ret = filters[idx].joy_button->get_button_index(); } else if (what == "pressed") { r_ret = filters[idx].joy_button->is_pressed(); } else { return false; } return true; } break; case Ref::SCREEN_TOUCH: { if (what == "finger_index") { r_ret = filters[idx].screen_touch.index; } else if (what == "pressed") { r_ret = filters[idx].screen_touch->is_pressed(); } else { return false; } return true; } break; case Ref::SCREEN_DRAG: { if (what == "finger_index") { r_ret = filters[idx].screen_drag.index; } else { return false; } return true; } break; case Ref::ACTION: { if (what == "action_name") { List pinfo; ProjectSettings::get_singleton()->get_property_list(&pinfo); int index = 1; for (List::Element *E = pinfo.front(); E; E = E->next()) { const PropertyInfo &pi = E->get(); if (!pi.name.begins_with("input/")) continue; if (filters[idx].action.action == index) { r_ret = pi.name.substr(pi.name.find("/") + 1, pi.name.length()); return true; } index++; } r_ret = "None"; //no index return false; } else if (what == "pressed") { r_ret = filters[idx].action->is_pressed(); return true; } } break; } } return false; } static const char *event_type_names[Ref::TYPE_MAX] = { "None", "Key", "MouseMotion", "MouseButton", "JoypadMotion", "JoypadButton", "ScreenTouch", "ScreenDrag", "Action" }; void VisualScriptInputFilter::_get_property_list(List *p_list) const { p_list->push_back(PropertyInfo(Variant::INT, "filter_count", PROPERTY_HINT_RANGE, "0,64")); String et; for (int i = 0; i < Ref::TYPE_MAX; i++) { if (i > 0) et += ","; et += event_type_names[i]; } String kc; String actions; for (int i = 0; i < filters.size(); i++) { String base = "filter_" + itos(i) + "/"; p_list->push_back(PropertyInfo(Variant::INT, base + "type", PROPERTY_HINT_ENUM, et)); p_list->push_back(PropertyInfo(Variant::INT, base + "device")); switch (filters[i].type) { case Ref::NONE: { } break; case Ref::KEY: { if (kc == String()) { int kcc = keycode_get_count(); kc = "None"; for (int i = 0; i < kcc; i++) { kc += ","; kc += String(keycode_get_name_by_index(i)); } } p_list->push_back(PropertyInfo(Variant::STRING, base + "scancode", PROPERTY_HINT_ENUM, kc)); p_list->push_back(PropertyInfo(Variant::STRING, base + "unicode")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "pressed")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "echo")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_alt")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_shift")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_ctrl")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_meta")); } break; case Ref::MOUSE_MOTION: { p_list->push_back(PropertyInfo(Variant::INT, base + "button_mask", PROPERTY_HINT_FLAGS, "Left,Right,Middle,WheelUp,WheelDown,WheelLeft,WheelRight")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_alt")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_shift")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_ctrl")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_meta")); } break; case Ref::MOUSE_BUTTON: { p_list->push_back(PropertyInfo(Variant::INT, base + "button_index", PROPERTY_HINT_ENUM, "Any,Left,Right,Middle,WheelUp,WheelDown,WheelLeft,WheelRight")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "pressed")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "doubleclicked")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_alt")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_shift")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_ctrl")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "mod_meta")); } break; case Ref::JOYPAD_MOTION: { p_list->push_back(PropertyInfo(Variant::INT, base + "axis_index")); p_list->push_back(PropertyInfo(Variant::INT, base + "mode", PROPERTY_HINT_ENUM, "Min,Max")); p_list->push_back(PropertyInfo(Variant::REAL, base + "threshold", PROPERTY_HINT_RANGE, "0,1,0.01")); } break; case Ref::JOYPAD_BUTTON: { p_list->push_back(PropertyInfo(Variant::INT, base + "button_index")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "pressed")); } break; case Ref::SCREEN_TOUCH: { p_list->push_back(PropertyInfo(Variant::INT, base + "finger_index")); p_list->push_back(PropertyInfo(Variant::BOOL, base + "pressed")); } break; case Ref::SCREEN_DRAG: { p_list->push_back(PropertyInfo(Variant::INT, base + "finger_index")); } break; case Ref::ACTION: { if (actions == String()) { actions = "None"; List pinfo; ProjectSettings::get_singleton()->get_property_list(&pinfo); Vector al; for (List::Element *E = pinfo.front(); E; E = E->next()) { const PropertyInfo &pi = E->get(); if (!pi.name.begins_with("input/")) continue; String name = pi.name.substr(pi.name.find("/") + 1, pi.name.length()); al.push_back(name); } for (int i = 0; i < al.size(); i++) { actions += ","; actions += al[i]; } } p_list->push_back(PropertyInfo(Variant::STRING, base + "action_name", PROPERTY_HINT_ENUM, actions)); p_list->push_back(PropertyInfo(Variant::BOOL, base + "pressed")); } break; } } } class VisualScriptNodeInstanceInputFilter : public VisualScriptNodeInstance { public: VisualScriptInstance *instance; Vector> filters; //virtual int get_working_memory_size() const { return 0; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return false; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { if (p_inputs[0]->get_type() != Variant::INPUT_EVENT) { r_error_str = "Input value not of type event"; r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD; return 0; } Ref event = *p_inputs[0]; for (int i = 0; i < filters.size(); i++) { const Ref &ie = filters[i]; if (ie.type != event.type) continue; bool match = false; switch (ie.type) { case Ref::NONE: { match = true; } break; case Ref::KEY: { InputEventKey k = ie.key; InputEventKey k2 = event.key; if (k->get_scancode() == 0 && k.unicode == 0 && k2->get_scancode() == 0 && k2.unicode == 0) { match = true; } else { if ((k->get_scancode() != 0 && k->get_scancode() == k2->get_scancode()) || (k.unicode != 0 && k.unicode == k2.unicode)) { //key valid if ( k->is_pressed() == k2->is_pressed() && k.echo == k2.echo && k.mod == k2.mod) { match = true; } } } } break; case Ref::MOUSE_MOTION: { InputEventMouseMotion mm = ie.mouse_motion; InputEventMouseMotion mm2 = event.mouse_motion; if (mm->get_button_mask() == mm2->get_button_mask() && mm.mod == mm2.mod) { match = true; } } break; case Ref::MOUSE_BUTTON: { InputEventMouseButton mb = ie.mouse_button; InputEventMouseButton mb2 = event.mouse_button; if (mb->get_button_index() == mb2->get_button_index() && mb->is_pressed() == mb2->is_pressed() && mb.doubleclick == mb2.doubleclick && mb.mod == mb2.mod) { match = true; } } break; case Ref::JOYPAD_MOTION: { InputEventJoypadMotion jm = ie.joy_motion; InputEventJoypadMotion jm2 = event.joy_motion; int axis = jm.axis >> 1; if (axis == jm2.axis) { if (jm.axis & 1) { //greater if (jm2.axis_value > jm.axis_value) { match = true; } } else { //less if (jm2.axis_value < -jm.axis_value) { match = true; } } } } break; case Ref::JOYPAD_BUTTON: { InputEventJoypadButton jb = ie.joy_button; InputEventJoypadButton jb2 = event.joy_button; if (jb->get_button_index() == jb2->get_button_index() && jb->is_pressed() == jb2->is_pressed()) { match = true; } } break; case Ref::SCREEN_TOUCH: { InputEventScreenTouch st = ie.screen_touch; InputEventScreenTouch st2 = event.screen_touch; if (st.index == st2.index && st->is_pressed() == st2->is_pressed()) { match = true; } } break; case Ref::SCREEN_DRAG: { InputEventScreenDrag sd = ie.screen_drag; InputEventScreenDrag sd2 = event.screen_drag; if (sd.index == sd2.index) { match = true; } } break; case Ref::ACTION: { InputEventAction ia = ie.action; InputEventAction ia2 = event.action; if (ia.action == ia2.action && ia->is_pressed() == ia2->is_pressed()) { match = true; } } break; } *p_outputs[0] = event; if (match) return i; //go through match output } return STEP_NO_ADVANCE_BIT; //none found, don't advance } }; VisualScriptNodeInstance *VisualScriptInputFilter::instance(VisualScriptInstance *p_instance) { VisualScriptNodeInstanceInputFilter *instance = memnew(VisualScriptNodeInstanceInputFilter); instance->instance = p_instance; instance->filters = filters; return instance; } VisualScriptInputFilter::VisualScriptInputFilter() { } #endif ////////////////////////////////////////// ////////////////TYPE CAST/////////// ////////////////////////////////////////// int VisualScriptTypeCast::get_output_sequence_port_count() const { return 2; } bool VisualScriptTypeCast::has_input_sequence_port() const { return true; } int VisualScriptTypeCast::get_input_value_port_count() const { return 1; } int VisualScriptTypeCast::get_output_value_port_count() const { return 1; } String VisualScriptTypeCast::get_output_sequence_port_text(int p_port) const { return p_port == 0 ? "yes" : "no"; } PropertyInfo VisualScriptTypeCast::get_input_value_port_info(int p_idx) const { return PropertyInfo(Variant::OBJECT, "instance"); } PropertyInfo VisualScriptTypeCast::get_output_value_port_info(int p_idx) const { return PropertyInfo(Variant::OBJECT, ""); } String VisualScriptTypeCast::get_caption() const { return "TypeCast"; } String VisualScriptTypeCast::get_text() const { if (script != String()) return "Is " + script.get_file() + "?"; else return "Is " + base_type + "?"; } void VisualScriptTypeCast::set_base_type(const StringName &p_type) { if (base_type == p_type) return; base_type = p_type; _change_notify(); ports_changed_notify(); } StringName VisualScriptTypeCast::get_base_type() const { return base_type; } void VisualScriptTypeCast::set_base_script(const String &p_path) { if (script == p_path) return; script = p_path; _change_notify(); ports_changed_notify(); } String VisualScriptTypeCast::get_base_script() const { return script; } VisualScriptTypeCast::TypeGuess VisualScriptTypeCast::guess_output_type(TypeGuess *p_inputs, int p_output) const { TypeGuess tg; tg.type = Variant::OBJECT; if (script != String()) { tg.script = ResourceLoader::load(script); } //if (!tg.script.is_valid()) { // tg.gdclass = base_type; //} return tg; } class VisualScriptNodeInstanceTypeCast : public VisualScriptNodeInstance { public: VisualScriptInstance *instance; StringName base_type; String script; //virtual int get_working_memory_size() const { return 0; } //virtual bool is_output_port_unsequenced(int p_idx) const { return false; } //virtual bool get_output_port_unsequenced(int p_idx,Variant* r_value,Variant* p_working_mem,String &r_error) const { return false; } virtual int step(const Variant **p_inputs, Variant **p_outputs, StartMode p_start_mode, Variant *p_working_mem, Variant::CallError &r_error, String &r_error_str) { Object *obj = *p_inputs[0]; *p_outputs[0] = Variant(); if (!obj) { r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD; r_error_str = "Instance is null"; return 0; } if (script != String()) { Ref