mirror of
				https://github.com/godotengine/godot.git
				synced 2025-10-25 18:54:43 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1962 lines
		
	
	
	
		
			46 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1962 lines
		
	
	
	
		
			46 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "visual_script_flow_control.h"
 | |
| #include "os/keyboard.h"
 | |
| #include "globals.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() {
 | |
| 
 | |
| 	ObjectTypeDB::bind_method(_MD("set_return_type","type"),&VisualScriptReturn::set_return_type);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_return_type"),&VisualScriptReturn::get_return_type);
 | |
| 	ObjectTypeDB::bind_method(_MD("set_enable_return_value","enable"),&VisualScriptReturn::set_enable_return_value);
 | |
| 	ObjectTypeDB::bind_method(_MD("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_value/enabled"),_SCS("set_enable_return_value"),_SCS("is_return_value_enabled"));
 | |
| 	ADD_PROPERTY(PropertyInfo(Variant::INT,"return_value/type",PROPERTY_HINT_ENUM,argt),_SCS("set_return_type"),_SCS("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<bool with_value>
 | |
| static Ref<VisualScriptNode> create_return_node(const String& p_name) {
 | |
| 
 | |
| 	Ref<VisualScriptReturn> 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() {
 | |
| 
 | |
| 	ObjectTypeDB::bind_method(_MD("set_steps","steps"),&VisualScriptSequence::set_steps);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_steps"),&VisualScriptSequence::get_steps);
 | |
| 
 | |
| 	ADD_PROPERTY(PropertyInfo(Variant::INT,"steps",PROPERTY_HINT_RANGE,"1,64,1"),_SCS("set_steps"),_SCS("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 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";
 | |
| 
 | |
| 	if (case_values[p_port].value.get_type()==Variant::NIL)
 | |
| 		return "null";
 | |
| 	return case_values[p_port].value;
 | |
| }
 | |
| 
 | |
| PropertyInfo VisualScriptSwitch::get_input_value_port_info(int p_idx) const{
 | |
| 
 | |
| 	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;
 | |
| 	Vector<Variant> case_values;
 | |
| 
 | |
| 	//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_values.size(); //exit
 | |
| 		}
 | |
| 
 | |
| 		for(int i=0;i<case_values.size();i++) {
 | |
| 
 | |
| 			if (*p_inputs[0]==case_values[i]) {
 | |
| 				return i|STEP_FLAG_PUSH_STACK_BIT;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return case_values.size();
 | |
| 	}
 | |
| 
 | |
| 
 | |
| };
 | |
| 
 | |
| VisualScriptNodeInstance* VisualScriptSwitch::instance(VisualScriptInstance* p_instance) {
 | |
| 
 | |
| 	VisualScriptNodeInstanceSwitch * instance = memnew(VisualScriptNodeInstanceSwitch );
 | |
| 	instance->instance=p_instance;
 | |
| 	instance->case_values.resize(case_values.size());
 | |
| 	for(int i=0;i<case_values.size();i++) {
 | |
| 		instance->case_values[i]=case_values[i].value;
 | |
| 	}
 | |
| 	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);
 | |
| 		String what = String(p_name).get_slice("/",2);
 | |
| 
 | |
| 		if (what=="type") {
 | |
| 			case_values[idx].type=Variant::Type(int(p_value));
 | |
| 			Variant::CallError ce;
 | |
| 			case_values[idx].value=Variant::construct(case_values[idx].type,NULL,0,ce);
 | |
| 			_change_notify();
 | |
| 			ports_changed_notify();
 | |
| 
 | |
| 			return true;
 | |
| 		}
 | |
| 
 | |
| 		if (what=="value") {
 | |
| 			case_values[idx].value=p_value;
 | |
| 			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);
 | |
| 		String what = String(p_name).get_slice("/",2);
 | |
| 
 | |
| 		if (what=="type") {
 | |
| 			r_ret=case_values[idx].type;
 | |
| 			return true;
 | |
| 		}
 | |
| 
 | |
| 		if (what=="value") {
 | |
| 			r_ret=case_values[idx].value;
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| 
 | |
| }
 | |
| void VisualScriptSwitch::_get_property_list( List<PropertyInfo> *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)+"/type",PROPERTY_HINT_ENUM,argt));
 | |
| 		p_list->push_back(PropertyInfo(case_values[i].type,"case/"+itos(i)+"/value"));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| void VisualScriptSwitch::_bind_methods() {
 | |
| 
 | |
| 
 | |
| }
 | |
| 
 | |
| VisualScriptSwitch::VisualScriptSwitch() {
 | |
| 
 | |
| 
 | |
| }
 | |
| 
 | |
| //////////////////////////////////////////
 | |
| ////////////////EVENT ACTION FILTER///////////
 | |
| //////////////////////////////////////////
 | |
| 
 | |
| 
 | |
| 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 InputEvent::NONE: {
 | |
| 			text="None";
 | |
| 		} break;
 | |
| 		case InputEvent::KEY: {
 | |
| 
 | |
| 			InputEventKey k = filters[p_port].key;
 | |
| 
 | |
| 			if (k.scancode==0 && k.unicode==0) {
 | |
| 				text="No Key";
 | |
| 			} else {
 | |
| 				if (k.scancode!=0) {
 | |
| 					text="KeyCode: "+keycode_get_string(k.scancode);
 | |
| 				} else if (k.unicode!=0) {
 | |
| 					text="Uniode: "+String::chr(k.unicode);
 | |
| 				}
 | |
| 
 | |
| 				if (k.pressed)
 | |
| 					text+=", Pressed";
 | |
| 				else
 | |
| 					text+=", Released";
 | |
| 
 | |
| 				if (k.echo)
 | |
| 					text+=", Echo";
 | |
| 				if (k.mod.alt)
 | |
| 					text="Alt+"+text;
 | |
| 				if (k.mod.shift)
 | |
| 					text="Shift+"+text;
 | |
| 				if (k.mod.control)
 | |
| 					text="Ctrl+"+text;
 | |
| 				if (k.mod.meta)
 | |
| 					text="Meta+"+text;
 | |
| 			}
 | |
| 
 | |
| 		} break;
 | |
| 		case InputEvent::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.button_mask&(1<<i)) {
 | |
| 					text=b.get_slice(",",i)+"+"+text;
 | |
| 				}
 | |
| 			}
 | |
| 			if (mm.mod.alt)
 | |
| 				text="Alt+"+text;
 | |
| 			if (mm.mod.shift)
 | |
| 				text="Shift+"+text;
 | |
| 			if (mm.mod.control)
 | |
| 				text="Ctrl+"+text;
 | |
| 			if (mm.mod.meta)
 | |
| 				text="Meta+"+text;
 | |
| 		} break;
 | |
| 		case InputEvent::MOUSE_BUTTON: {
 | |
| 
 | |
| 			InputEventMouseButton mb = filters[p_port].mouse_button;
 | |
| 
 | |
| 			String b = "Any,Left,Right,Middle,WheelUp,WheelDown,WheelLeft,WheelRight";
 | |
| 
 | |
| 			text=b.get_slice(",",mb.button_index)+" Mouse Button";
 | |
| 
 | |
| 			if (mb.pressed)
 | |
| 				text+=", Pressed";
 | |
| 			else
 | |
| 				text+=", Released";
 | |
| 
 | |
| 			if (mb.doubleclick)
 | |
| 				text+=", DblClick";
 | |
| 			if (mb.mod.alt)
 | |
| 				text="Alt+"+text;
 | |
| 			if (mb.mod.shift)
 | |
| 				text="Shift+"+text;
 | |
| 			if (mb.mod.control)
 | |
| 				text="Ctrl+"+text;
 | |
| 			if (mb.mod.meta)
 | |
| 				text="Meta+"+text;
 | |
| 
 | |
| 
 | |
| 		} break;
 | |
| 		case InputEvent::JOYSTICK_MOTION: {
 | |
| 
 | |
| 			InputEventJoystickMotion 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 InputEvent::JOYSTICK_BUTTON: {
 | |
| 			InputEventJoystickButton jb = filters[p_port].joy_button;
 | |
| 
 | |
| 			text="JoyButton "+itos(jb.button_index);
 | |
| 			if (jb.pressed)
 | |
| 				text+=", Pressed";
 | |
| 			else
 | |
| 				text+=", Released";
 | |
| 		} break;
 | |
| 		case InputEvent::SCREEN_TOUCH: {
 | |
| 			InputEventScreenTouch sd = filters[p_port].screen_touch;
 | |
| 
 | |
| 			text="Touch Finger "+itos(sd.index);
 | |
| 			if (sd.pressed)
 | |
| 				text+=", Pressed";
 | |
| 			else
 | |
| 				text+=", Released";
 | |
| 		} break;
 | |
| 		case InputEvent::SCREEN_DRAG: {
 | |
| 			InputEventScreenDrag sd = filters[p_port].screen_drag;
 | |
| 			text="Drag Finger "+itos(sd.index);
 | |
| 		} break;
 | |
| 		case InputEvent::ACTION: {
 | |
| 
 | |
| 
 | |
| 			List<PropertyInfo> pinfo;
 | |
| 			Globals::get_singleton()->get_property_list(&pinfo);
 | |
| 			int index=1;
 | |
| 
 | |
| 			text="No Action";
 | |
| 			for(List<PropertyInfo>::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.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]=InputEvent();
 | |
| 			filters[idx].type=InputEvent::Type(int(p_value));
 | |
| 			if (filters[idx].type==InputEvent::JOYSTICK_MOTION) {
 | |
| 				filters[idx].joy_motion.axis_value=0.5; //for treshold
 | |
| 			} else if (filters[idx].type==InputEvent::KEY) {
 | |
| 				filters[idx].key.pressed=true; //put these as true to make it more user friendly
 | |
| 			} else if (filters[idx].type==InputEvent::MOUSE_BUTTON) {
 | |
| 				filters[idx].mouse_button.pressed=true;
 | |
| 			} else if (filters[idx].type==InputEvent::JOYSTICK_BUTTON) {
 | |
| 				filters[idx].joy_button.pressed=true;
 | |
| 			} else if (filters[idx].type==InputEvent::SCREEN_TOUCH) {
 | |
| 				filters[idx].screen_touch.pressed=true;
 | |
| 			} else if (filters[idx].type==InputEvent::ACTION) {
 | |
| 				filters[idx].action.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 InputEvent::KEY: {
 | |
| 
 | |
| 				if (what=="scancode") {
 | |
| 					String sc = p_value;
 | |
| 					if (sc==String()) {
 | |
| 						filters[idx].key.scancode=0;
 | |
| 					} else {
 | |
| 						filters[idx].key.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].key.pressed=p_value;
 | |
| 				} else if (what=="echo") {
 | |
| 
 | |
| 					filters[idx].key.echo=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_alt") {
 | |
| 					filters[idx].key.mod.alt=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_shift") {
 | |
| 					filters[idx].key.mod.shift=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_ctrl") {
 | |
| 					filters[idx].key.mod.control=p_value;
 | |
| 
 | |
| 				} else  if (what=="mod_meta") {
 | |
| 					filters[idx].key.mod.meta=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				ports_changed_notify();
 | |
| 
 | |
| 				return true;
 | |
| 			} break;
 | |
| 			case InputEvent::MOUSE_MOTION: {
 | |
| 
 | |
| 
 | |
| 				if (what=="button_mask") {
 | |
| 					filters[idx].mouse_motion.button_mask=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_alt") {
 | |
| 					filters[idx].mouse_motion.mod.alt=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_shift") {
 | |
| 					filters[idx].mouse_motion.mod.shift=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_ctrl") {
 | |
| 					filters[idx].mouse_motion.mod.control=p_value;
 | |
| 
 | |
| 				} else  if (what=="mod_meta") {
 | |
| 					filters[idx].mouse_motion.mod.meta=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 
 | |
| 				ports_changed_notify();
 | |
| 				return true;
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::MOUSE_BUTTON: {
 | |
| 
 | |
| 				if (what=="button_index") {
 | |
| 					filters[idx].mouse_button.button_index=p_value;
 | |
| 				} else if (what=="pressed") {
 | |
| 					filters[idx].mouse_button.pressed=p_value;
 | |
| 				} else if (what=="doubleclicked") {
 | |
| 					filters[idx].mouse_button.doubleclick=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_alt") {
 | |
| 					filters[idx].mouse_button.mod.alt=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_shift") {
 | |
| 					filters[idx].mouse_button.mod.shift=p_value;
 | |
| 
 | |
| 				} else if (what=="mod_ctrl") {
 | |
| 					filters[idx].mouse_button.mod.control=p_value;
 | |
| 
 | |
| 				} else  if (what=="mod_meta") {
 | |
| 					filters[idx].mouse_button.mod.meta=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				ports_changed_notify();
 | |
| 				return true;
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::JOYSTICK_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=="treshold") {
 | |
| 					filters[idx].joy_motion.axis_value=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				ports_changed_notify();
 | |
| 				return true;
 | |
| 
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::JOYSTICK_BUTTON: {
 | |
| 
 | |
| 				if (what=="button_index") {
 | |
| 					filters[idx].joy_button.button_index=p_value;
 | |
| 				} else if (what=="pressed") {
 | |
| 					filters[idx].joy_button.pressed=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				ports_changed_notify();
 | |
| 				return true;
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::SCREEN_TOUCH: {
 | |
| 
 | |
| 				if (what=="finger_index") {
 | |
| 					filters[idx].screen_touch.index=p_value;
 | |
| 				} else if (what=="pressed") {
 | |
| 					filters[idx].screen_touch.pressed=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				ports_changed_notify();
 | |
| 				return true;
 | |
| 			} break;
 | |
| 			case InputEvent::SCREEN_DRAG: {
 | |
| 				if (what=="finger_index") {
 | |
| 					filters[idx].screen_drag.index=p_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				ports_changed_notify();
 | |
| 				return true;
 | |
| 			} break;
 | |
| 			case InputEvent::ACTION: {
 | |
| 
 | |
| 
 | |
| 				if (what=="action_name") {
 | |
| 
 | |
| 					List<PropertyInfo> pinfo;
 | |
| 					Globals::get_singleton()->get_property_list(&pinfo);
 | |
| 					int index=1;
 | |
| 
 | |
| 					for(List<PropertyInfo>::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.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 InputEvent::KEY: {
 | |
| 
 | |
| 				if (what=="scancode") {
 | |
| 					if (filters[idx].key.scancode==0)
 | |
| 						r_ret=String();
 | |
| 					else {
 | |
| 
 | |
| 						r_ret=keycode_get_string(filters[idx].key.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].key.pressed;
 | |
| 				} else if (what=="echo") {
 | |
| 
 | |
| 					r_ret=filters[idx].key.echo;
 | |
| 
 | |
| 				} else if (what=="mod_alt") {
 | |
| 					r_ret=filters[idx].key.mod.alt;
 | |
| 
 | |
| 				} else if (what=="mod_shift") {
 | |
| 					r_ret=filters[idx].key.mod.shift;
 | |
| 
 | |
| 				} else if (what=="mod_ctrl") {
 | |
| 					r_ret=filters[idx].key.mod.control;
 | |
| 
 | |
| 				} else  if (what=="mod_meta") {
 | |
| 					r_ret=filters[idx].key.mod.meta;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 
 | |
| 				return true;
 | |
| 			} break;
 | |
| 			case InputEvent::MOUSE_MOTION: {
 | |
| 
 | |
| 
 | |
| 				if (what=="button_mask") {
 | |
| 					r_ret=filters[idx].mouse_motion.button_mask;
 | |
| 
 | |
| 				} else if (what=="mod_alt") {
 | |
| 					r_ret=filters[idx].mouse_motion.mod.alt;
 | |
| 
 | |
| 				} else if (what=="mod_shift") {
 | |
| 					r_ret=filters[idx].mouse_motion.mod.shift;
 | |
| 
 | |
| 				} else if (what=="mod_ctrl") {
 | |
| 					r_ret=filters[idx].mouse_motion.mod.control;
 | |
| 
 | |
| 				} else  if (what=="mod_meta") {
 | |
| 					r_ret=filters[idx].mouse_motion.mod.meta;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 
 | |
| 				return true;
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::MOUSE_BUTTON: {
 | |
| 
 | |
| 				if (what=="button_index") {
 | |
| 					r_ret=filters[idx].mouse_button.button_index;
 | |
| 				} else if (what=="pressed") {
 | |
| 					r_ret=filters[idx].mouse_button.pressed;
 | |
| 				} else if (what=="doubleclicked") {
 | |
| 					r_ret=filters[idx].mouse_button.doubleclick;
 | |
| 
 | |
| 				} else if (what=="mod_alt") {
 | |
| 					r_ret=filters[idx].mouse_button.mod.alt;
 | |
| 
 | |
| 				} else if (what=="mod_shift") {
 | |
| 					r_ret=filters[idx].mouse_button.mod.shift;
 | |
| 
 | |
| 				} else if (what=="mod_ctrl") {
 | |
| 					r_ret=filters[idx].mouse_button.mod.control;
 | |
| 
 | |
| 				} else  if (what=="mod_meta") {
 | |
| 					r_ret=filters[idx].mouse_button.mod.meta;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				return true;
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::JOYSTICK_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=="treshold") {
 | |
| 					r_ret=filters[idx].joy_motion.axis_value;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				return true;
 | |
| 
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::JOYSTICK_BUTTON: {
 | |
| 
 | |
| 				if (what=="button_index") {
 | |
| 					r_ret=filters[idx].joy_button.button_index;
 | |
| 				} else if (what=="pressed") {
 | |
| 					r_ret=filters[idx].joy_button.pressed;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				return true;
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::SCREEN_TOUCH: {
 | |
| 
 | |
| 				if (what=="finger_index") {
 | |
| 					r_ret=filters[idx].screen_touch.index;
 | |
| 				} else if (what=="pressed") {
 | |
| 					r_ret=filters[idx].screen_touch.pressed;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				return true;
 | |
| 			} break;
 | |
| 			case InputEvent::SCREEN_DRAG: {
 | |
| 				if (what=="finger_index") {
 | |
| 					r_ret=filters[idx].screen_drag.index;
 | |
| 				} else {
 | |
| 					return false;
 | |
| 				}
 | |
| 				return true;
 | |
| 			} break;
 | |
| 			case InputEvent::ACTION: {
 | |
| 
 | |
| 
 | |
| 				if (what=="action_name") {
 | |
| 
 | |
| 					List<PropertyInfo> pinfo;
 | |
| 					Globals::get_singleton()->get_property_list(&pinfo);
 | |
| 					int index=1;
 | |
| 
 | |
| 					for(List<PropertyInfo>::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.pressed;
 | |
| 					return true;
 | |
| 				}
 | |
| 
 | |
| 
 | |
| 			} break;
 | |
| 
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| static const char* event_type_names[InputEvent::TYPE_MAX]={
 | |
| 	"None",
 | |
| 	"Key",
 | |
| 	"MouseMotion",
 | |
| 	"MouseButton",
 | |
| 	"JoystickMotion",
 | |
| 	"JoystickButton",
 | |
| 	"ScreenTouch",
 | |
| 	"ScreenDrag",
 | |
| 	"Action"
 | |
| };
 | |
| 
 | |
| void VisualScriptInputFilter::_get_property_list( List<PropertyInfo> *p_list) const {
 | |
| 
 | |
| 	p_list->push_back(PropertyInfo(Variant::INT,"filter_count",PROPERTY_HINT_RANGE,"0,64"));
 | |
| 
 | |
| 	String et;
 | |
| 	for(int i=0;i<InputEvent::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 InputEvent::NONE: {
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::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 InputEvent::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 InputEvent::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 InputEvent::JOYSTICK_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+"treshold",PROPERTY_HINT_RANGE,"0,1,0.01"));
 | |
| 			} break;
 | |
| 			case InputEvent::JOYSTICK_BUTTON: {
 | |
| 				p_list->push_back(PropertyInfo(Variant::INT,base+"button_index"));
 | |
| 				p_list->push_back(PropertyInfo(Variant::BOOL,base+"pressed"));
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::SCREEN_TOUCH: {
 | |
| 				p_list->push_back(PropertyInfo(Variant::INT,base+"finger_index"));
 | |
| 				p_list->push_back(PropertyInfo(Variant::BOOL,base+"pressed"));
 | |
| 
 | |
| 			} break;
 | |
| 			case InputEvent::SCREEN_DRAG: {
 | |
| 				p_list->push_back(PropertyInfo(Variant::INT,base+"finger_index"));
 | |
| 			} break;
 | |
| 			case InputEvent::ACTION: {
 | |
| 
 | |
| 
 | |
| 
 | |
| 				if (actions==String()) {
 | |
| 
 | |
| 					actions="None";
 | |
| 
 | |
| 					List<PropertyInfo> pinfo;
 | |
| 					Globals::get_singleton()->get_property_list(&pinfo);
 | |
| 					Vector<String> al;
 | |
| 
 | |
| 					for(List<PropertyInfo>::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<InputEvent> 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;
 | |
| 		}
 | |
| 
 | |
| 		InputEvent event = *p_inputs[0];
 | |
| 
 | |
| 
 | |
| 		for(int i=0;i<filters.size();i++) {
 | |
| 
 | |
| 			const InputEvent &ie = filters[i];
 | |
| 			if (ie.type!=event.type)
 | |
| 				continue;
 | |
| 
 | |
| 			bool match=false;
 | |
| 
 | |
| 			switch(ie.type) {
 | |
| 
 | |
| 				case InputEvent::NONE: {
 | |
| 
 | |
| 					match=true;
 | |
| 				} break;
 | |
| 				case InputEvent::KEY: {
 | |
| 
 | |
| 					InputEventKey k = ie.key;
 | |
| 					InputEventKey k2 = event.key;
 | |
| 
 | |
| 					if (k.scancode==0 && k.unicode==0 && k2.scancode==0 && k2.unicode==0) {
 | |
| 						match=true;
 | |
| 
 | |
| 					} else {
 | |
| 
 | |
| 						if ( (k.scancode!=0 && k.scancode==k2.scancode) || (k.unicode!=0 && k.unicode==k2.unicode)) {
 | |
| 							//key valid
 | |
| 
 | |
| 							if (
 | |
| 								k.pressed==k2.pressed &&
 | |
| 								k.echo==k2.echo &&
 | |
| 								k.mod == k2.mod
 | |
| 									) {
 | |
| 								match=true;
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 					}
 | |
| 
 | |
| 				} break;
 | |
| 				case InputEvent::MOUSE_MOTION: {
 | |
| 					InputEventMouseMotion mm = ie.mouse_motion;
 | |
| 					InputEventMouseMotion mm2 = event.mouse_motion;
 | |
| 
 | |
| 					if (	mm.button_mask==mm2.button_mask &&
 | |
| 						mm.mod==mm2.mod
 | |
| 						) {
 | |
| 						match=true;
 | |
| 					}
 | |
| 
 | |
| 				} break;
 | |
| 				case InputEvent::MOUSE_BUTTON: {
 | |
| 
 | |
| 					InputEventMouseButton mb = ie.mouse_button;
 | |
| 					InputEventMouseButton mb2 = event.mouse_button;
 | |
| 
 | |
| 					if (	mb.button_index==mb2.button_index &&
 | |
| 						mb.pressed==mb2.pressed &&
 | |
| 						mb.doubleclick==mb2.doubleclick &&
 | |
| 						mb.mod==mb2.mod) {
 | |
| 						match=true;
 | |
| 					}
 | |
| 
 | |
| 
 | |
| 				} break;
 | |
| 				case InputEvent::JOYSTICK_MOTION: {
 | |
| 
 | |
| 					InputEventJoystickMotion jm = ie.joy_motion;
 | |
| 					InputEventJoystickMotion 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 InputEvent::JOYSTICK_BUTTON: {
 | |
| 					InputEventJoystickButton jb = ie.joy_button;
 | |
| 					InputEventJoystickButton jb2 = event.joy_button;
 | |
| 
 | |
| 					if (	jb.button_index==jb2.button_index &&
 | |
| 						jb.pressed == jb2.pressed
 | |
| 							) {
 | |
| 						match=true;
 | |
| 					}
 | |
| 				} break;
 | |
| 				case InputEvent::SCREEN_TOUCH: {
 | |
| 					InputEventScreenTouch st = ie.screen_touch;
 | |
| 					InputEventScreenTouch st2 = event.screen_touch;
 | |
| 
 | |
| 					if (	st.index==st2.index &&
 | |
| 						st.pressed==st2.pressed) {
 | |
| 						match=true;
 | |
| 					}
 | |
| 
 | |
| 				} break;
 | |
| 				case InputEvent::SCREEN_DRAG: {
 | |
| 					InputEventScreenDrag sd = ie.screen_drag;
 | |
| 					InputEventScreenDrag sd2 = event.screen_drag;
 | |
| 
 | |
| 					if (sd.index==sd2.index) {
 | |
| 						match=true;
 | |
| 					}
 | |
| 				} break;
 | |
| 				case InputEvent::ACTION: {
 | |
| 
 | |
| 					InputEventAction ia = ie.action;
 | |
| 					InputEventAction ia2 = event.action;
 | |
| 
 | |
| 					if (	ia.action==ia2.action &&
 | |
| 						ia.pressed==ia2.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() {
 | |
| 
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| //////////////////////////////////////////
 | |
| ////////////////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;
 | |
| }
 | |
| 
 | |
| 
 | |
| 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<Script> obj_script = obj->get_script();
 | |
| 			if (!obj_script.is_valid()) {
 | |
| 				return 1; //well, definitely not the script because object we got has no script.
 | |
| 			}
 | |
| 
 | |
| 			if (!ResourceCache::has(script)) {
 | |
| 				//if the script is not in use by anyone, we can safely assume whathever we got is not casting to it.
 | |
| 				return 1;
 | |
| 			}
 | |
| 			Ref<Script> cast_script = Ref<Resource>(ResourceCache::get(script));
 | |
| 			if (!cast_script.is_valid()) {
 | |
| 				r_error.error=Variant::CallError::CALL_ERROR_INVALID_METHOD;
 | |
| 				r_error_str="Script path is not a script: "+script;
 | |
| 				return 1;
 | |
| 			}
 | |
| 
 | |
| 			while(obj_script.is_valid()) {
 | |
| 
 | |
| 				if (cast_script==obj_script) {
 | |
| 					*p_outputs[0]=*p_inputs[0]; //copy
 | |
| 					return 0; // it is the script, yey
 | |
| 				}
 | |
| 
 | |
| 				obj_script=obj_script->get_base_script();
 | |
| 			}
 | |
| 
 | |
| 			return 1; //not found sorry
 | |
| 		}
 | |
| 
 | |
| 		if (ObjectTypeDB::is_type(obj->get_type_name(),base_type)) {
 | |
| 			*p_outputs[0]=*p_inputs[0]; //copy
 | |
| 			return 0;
 | |
| 		} else
 | |
| 			return 1;
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 
 | |
| };
 | |
| 
 | |
| VisualScriptNodeInstance* VisualScriptTypeCast::instance(VisualScriptInstance* p_instance) {
 | |
| 
 | |
| 	VisualScriptNodeInstanceTypeCast * instance = memnew(VisualScriptNodeInstanceTypeCast );
 | |
| 	instance->instance=p_instance;
 | |
| 	instance->base_type=base_type;
 | |
| 	instance->script=script;
 | |
| 	return instance;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| void VisualScriptTypeCast::_bind_methods() {
 | |
| 
 | |
| 	ObjectTypeDB::bind_method(_MD("set_base_type","type"),&VisualScriptTypeCast::set_base_type);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_base_type"),&VisualScriptTypeCast::get_base_type);
 | |
| 
 | |
| 	ObjectTypeDB::bind_method(_MD("set_base_script","path"),&VisualScriptTypeCast::set_base_script);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_base_script"),&VisualScriptTypeCast::get_base_script);
 | |
| 
 | |
| 
 | |
| 	List<String> script_extensions;
 | |
| 	for(int i=0;i>ScriptServer::get_language_count();i++) {
 | |
| 		ScriptServer::get_language(i)->get_recognized_extensions(&script_extensions);
 | |
| 	}
 | |
| 
 | |
| 	String script_ext_hint;
 | |
| 	for (List<String>::Element *E=script_extensions.front();E;E=E->next()) {
 | |
| 		if (script_ext_hint!=String())
 | |
| 			script_ext_hint+=",";
 | |
| 		script_ext_hint+="*."+E->get();
 | |
| 	}
 | |
| 
 | |
| 	ADD_PROPERTY(PropertyInfo(Variant::STRING,"function/base_type",PROPERTY_HINT_TYPE_STRING,"Object"),_SCS("set_base_type"),_SCS("get_base_type"));
 | |
| 	ADD_PROPERTY(PropertyInfo(Variant::STRING,"property/base_script",PROPERTY_HINT_FILE,script_ext_hint),_SCS("set_base_script"),_SCS("get_base_script"));
 | |
| 
 | |
| }
 | |
| 
 | |
| VisualScriptTypeCast::VisualScriptTypeCast() {
 | |
| 
 | |
| 	base_type="Object";
 | |
| }
 | |
| 
 | |
| 
 | |
| void register_visual_script_flow_control_nodes() {
 | |
| 
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/return",create_return_node<false>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/return_with_value",create_return_node<true>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/condition",create_node_generic<VisualScriptCondition>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/while",create_node_generic<VisualScriptWhile>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/iterator",create_node_generic<VisualScriptIterator>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/sequence",create_node_generic<VisualScriptSequence>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/switch",create_node_generic<VisualScriptSwitch>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/input_filter",create_node_generic<VisualScriptInputFilter>);
 | |
| 	VisualScriptLanguage::singleton->add_register_func("flow_control/type_cast",create_node_generic<VisualScriptTypeCast>);
 | |
| 
 | |
| 
 | |
| 
 | |
| }
 | 
