Răsfoiți Sursa

Merge pull request #3814 from est31/iterators_for_for

Add Iterators and use them for for
Rémi Verschelde 9 ani în urmă
părinte
comite
adf5056889

+ 169 - 0
core/range_iterator.cpp

@@ -0,0 +1,169 @@
+/*************************************************************************/
+/*  range_iterator.cpp                                                   */
+/*************************************************************************/
+/*                       This file is part of:                           */
+/*                           GODOT ENGINE                                */
+/*                    http://www.godotengine.org                         */
+/*************************************************************************/
+/* Copyright (c) 2007-2016 Juan Linietsky, Ariel Manzur.                 */
+/*                                                                       */
+/* 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 "range_iterator.h"
+#include "object_type_db.h"
+
+void RangeIterator::_bind_methods() {
+	ObjectTypeDB::bind_method(_MD("_iter_init","arg"),&RangeIterator::_iter_init);
+	ObjectTypeDB::bind_method(_MD("_iter_next","arg"),&RangeIterator::_iter_next);
+	ObjectTypeDB::bind_method(_MD("_iter_get","arg"),&RangeIterator::_iter_get);
+	ObjectTypeDB::bind_method(_MD("is_finished"),&RangeIterator::is_finished);
+	ObjectTypeDB::bind_method(_MD("to_array"),&RangeIterator::to_array);
+	ObjectTypeDB::bind_method(_MD("set_range","arg1","arg2","arg3"),&RangeIterator::_set_range,DEFVAL(Variant()),DEFVAL(Variant()));
+}
+
+bool RangeIterator::_iter_init(Variant arg) {
+	return !is_finished();
+}
+
+bool RangeIterator::_iter_next(Variant arg) {
+	current += step;
+	return !is_finished();
+}
+
+Variant RangeIterator::_iter_get(Variant arg) {
+	return Variant(current);
+}
+
+bool RangeIterator::is_finished() {
+	if(step > 0)
+	{
+		return current >= stop;
+	}
+	else
+	{
+		return current <= stop;
+	}
+}
+
+Array RangeIterator::to_array() {
+	if (step==0) {
+		ERR_EXPLAIN("step is zero!");
+		ERR_FAIL_V(Array());
+	}
+
+	Array arr(true);
+	if (current >= stop && step > 0) {
+		return arr;
+	}
+	if (current <= stop && step < 0) {
+		return arr;
+	}
+
+	//calculate how many
+	int count=0;
+	if (step > 0) {
+		count=((stop-current-1)/step)+1;
+	} else {
+		count=((current-stop-1)/-step)+1;
+	}
+
+	arr.resize(count);
+
+	if (step > 0) {
+		int idx=0;
+		for(int i=current;i<stop;i+=step) {
+			arr[idx++]=i;
+		}
+	} else {
+		int idx=0;
+		for(int i=current;i>stop;i+=step) {
+			arr[idx++]=i;
+		}
+	}
+
+	return arr;
+}
+
+void RangeIterator::set_range(int stop) {
+	this->current = 0;
+	this->stop = stop;
+	this->step = (stop > 0)?(1):(-1);
+}
+
+void RangeIterator::set_range(int start, int stop) {
+	this->current = start;
+	this->stop = stop;
+	this->step = (stop > start)?(1):(-1);
+}
+
+void RangeIterator::set_range(int start, int stop, int step) {
+	if(step == 0)
+	{
+		ERR_EXPLAIN("step is zero!");
+		ERR_FAIL();
+	}
+
+	this->current = start;
+	this->stop = stop;
+	this->step = step;
+}
+
+Ref<RangeIterator> RangeIterator::_set_range(Variant arg1, Variant arg2, Variant arg3)
+{
+	bool valid = true;
+	if(arg1.get_type() == Variant::INT)
+	{
+		if(arg2.get_type() == Variant::INT)
+		{
+			if(arg3.get_type() == Variant::INT) set_range((int)arg1, (int)arg2, (int)arg3); // (start, end, step)
+			else if(arg3.get_type() == Variant::NIL) set_range((int)arg1, (int)arg2); // (start, end)
+			else valid = false;
+		}
+		else if(arg2.get_type() == Variant::NIL) set_range((int)arg1); // (end)
+		else valid = false;
+	}
+	else valid = false;
+
+	if(!valid)
+	{
+		ERR_EXPLAIN("Invalid type in function 'set_range' in base 'RangeIterator'. Expected 1, 2, or 3 ints.");
+		ERR_FAIL_V(Ref<RangeIterator>());
+	}
+	return Ref<RangeIterator>(this);
+}
+
+RangeIterator::RangeIterator() {
+	current = 0;
+	stop = 0;
+	step = 0;
+}
+
+RangeIterator::RangeIterator(int stop) {
+	set_range(stop);
+}
+
+RangeIterator::RangeIterator(int start, int stop) {
+	set_range(start, stop);
+}
+
+RangeIterator::RangeIterator(int start, int stop, int step) {
+	set_range(start, stop, step);
+}

+ 72 - 0
core/range_iterator.h

@@ -0,0 +1,72 @@
+/*************************************************************************/
+/*  range_iterator.h                                                     */
+/*************************************************************************/
+/*                       This file is part of:                           */
+/*                           GODOT ENGINE                                */
+/*                    http://www.godotengine.org                         */
+/*************************************************************************/
+/* Copyright (c) 2007-2016 Juan Linietsky, Ariel Manzur.                 */
+/*                                                                       */
+/* 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.                */
+/*************************************************************************/
+#ifndef RANGE_ITERATOR_H
+#define RANGE_ITERATOR_H
+
+#include "reference.h"
+#include "variant.h"
+#include "array.h"
+
+class RangeIterator : public Reference
+{
+protected:
+	OBJ_TYPE( RangeIterator, Reference );
+
+	static void _bind_methods();
+
+private:
+	int current;
+	int stop;
+	int step;
+
+	bool _iter_init(Variant arg);
+	bool _iter_next(Variant arg);
+	Variant _iter_get(Variant arg);
+
+public:
+
+	bool is_finished();
+
+	Array to_array();
+
+	void set_range(int stop);
+	void set_range(int start, int stop);
+	void set_range(int start, int stop, int step);
+
+	Ref<RangeIterator> _set_range(Variant arg1, Variant arg2 = Variant(), Variant arg3 = Variant());
+
+	void _init(Variant arg1, Variant arg2, Variant arg3);
+
+	RangeIterator();
+	RangeIterator(int stop);
+	RangeIterator(int start, int stop);
+	RangeIterator(int start, int stop, int step);
+};
+
+#endif // RANGE_ITERATOR_H

+ 3 - 0
core/register_core_types.cpp

@@ -52,6 +52,7 @@
 #include "func_ref.h"
 #include "input_map.h"
 #include "undo_redo.h"
+#include "range_iterator.h"
 
 #ifdef XML_ENABLED
 static ResourceFormatSaverXML *resource_saver_xml=NULL;
@@ -130,6 +131,8 @@ void register_core_types() {
 	ObjectTypeDB::register_type<Translation>();
 	ObjectTypeDB::register_type<PHashTranslation>();
 	ObjectTypeDB::register_type<UndoRedo>();
+	ObjectTypeDB::register_type<RangeIterator>();
+
 	ObjectTypeDB::register_type<HTTPClient>();
 
 	ObjectTypeDB::register_virtual_type<ResourceInteractiveLoader>();

+ 43 - 0
doc/base/classes.xml

@@ -556,6 +556,15 @@
 			Return an array with the given range. Range can be 1 argument N (0 to N-1), two arguments (initial, final-1) or three arguments (initial,final-1,increment).
 			</description>
 		</method>
+		<method name="xrange">
+			<return type="Object">
+			</return>
+			<argument index="0" name="..." type="Variant">
+			</argument>
+			<description>
+			Return an iterator over the given range. Range can be 1 argument N (0 to N-1), two arguments (initial, final-1) or three arguments (initial,final-1,increment).
+			</description>
+		</method>
 		<method name="load">
 			<return type="Resource">
 			</return>
@@ -27597,6 +27606,40 @@ This method controls whether the position between two cached points is interpola
 	<constants>
 	</constants>
 </class>
+<class name="RangeIterator" inherits="Reference" category="Core">
+	<brief_description>
+	</brief_description>
+	<description>
+	</description>
+	<methods>
+		<method name="is_finished">
+			<return type="bool">
+			</return>
+			<description>
+			</description>
+		</method>
+		<method name="to_array">
+			<return type="Array">
+			</return>
+			<description>
+			</description>
+		</method>
+		<method name="set_range">
+			<return type="Object">
+			</return>
+			<argument index="0" name="arg1" type="var">
+			</argument>
+			<argument index="1" name="arg2" type="var" default="NULL">
+			</argument>
+			<argument index="2" name="arg3" type="var" default="NULL">
+			</argument>
+			<description>
+			</description>
+		</method>
+	</methods>
+	<constants>
+	</constants>
+</class>
 <class name="RawArray" category="Built-In Types">
 	<brief_description>
 	Raw byte array.

+ 83 - 0
modules/gdscript/gd_functions.cpp

@@ -32,6 +32,7 @@
 #include "reference.h"
 #include "gd_script.h"
 #include "func_ref.h"
+#include "range_iterator.h"
 #include "os/os.h"
 #include "variant_parser.h"
 #include "io/marshalls.h"
@@ -98,6 +99,7 @@ const char *GDFunctions::get_func_name(Function p_func) {
 		"var2bytes",
 		"bytes2var",
 		"range",
+		"xrange",
 		"load",
 		"inst2dict",
 		"dict2inst",
@@ -815,6 +817,81 @@ void GDFunctions::call(Function p_func,const Variant **p_args,int p_arg_count,Va
 				} break;
 			}
 
+		} break;
+		case GEN_XRANGE: {
+
+			switch(p_arg_count) {
+				case 0: {
+					r_error.error=Variant::CallError::CALL_ERROR_TOO_FEW_ARGUMENTS;
+					r_error.argument=1;
+				} break;
+				case 1: {
+
+					VALIDATE_ARG_NUM(0);
+
+					int count=*p_args[0];
+
+					Ref<RangeIterator> itr = Ref<RangeIterator>( memnew(RangeIterator) );
+					if (!*itr) {
+						ERR_EXPLAIN("Couldn't allocate iterator!");
+						r_error.error=Variant::CallError::CALL_ERROR_INVALID_METHOD;
+						ERR_FAIL();
+					}
+					(*itr)->set_range(count);
+					r_ret=Variant(itr);
+					return;
+				} break;
+				case 2: {
+
+					VALIDATE_ARG_NUM(0);
+					VALIDATE_ARG_NUM(1);
+
+					int from=*p_args[0];
+					int to=*p_args[1];
+
+					Ref<RangeIterator> itr = Ref<RangeIterator>( memnew(RangeIterator) );
+					if (!*itr) {
+						ERR_EXPLAIN("Couldn't allocate iterator!");
+						r_error.error=Variant::CallError::CALL_ERROR_INVALID_METHOD;
+						ERR_FAIL();
+					}
+					(*itr)->set_range(from, to);
+					r_ret=Variant(itr);
+					return;
+				} break;
+				case 3: {
+
+					VALIDATE_ARG_NUM(0);
+					VALIDATE_ARG_NUM(1);
+					VALIDATE_ARG_NUM(2);
+
+					int from=*p_args[0];
+					int to=*p_args[1];
+					int incr=*p_args[2];
+
+					if (incr==0) {
+						ERR_EXPLAIN("step argument is zero!");
+						r_error.error=Variant::CallError::CALL_ERROR_INVALID_METHOD;
+						ERR_FAIL();
+					}
+
+					Ref<RangeIterator> itr = Ref<RangeIterator>( memnew(RangeIterator) );
+					if (!*itr) {
+						ERR_EXPLAIN("Couldn't allocate iterator!");
+						r_error.error=Variant::CallError::CALL_ERROR_INVALID_METHOD;
+						ERR_FAIL();
+					}
+					(*itr)->set_range(from, to, incr);
+					r_ret=Variant(itr);
+					return;
+				} break;
+				default: {
+
+					r_error.error=Variant::CallError::CALL_ERROR_TOO_MANY_ARGUMENTS;
+					r_error.argument=3;
+				} break;
+			}
+
 		} break;
 		case RESOURCE_LOAD: {
 			VALIDATE_ARG_COUNT(1);
@@ -1433,6 +1510,12 @@ MethodInfo GDFunctions::get_info(Function p_func) {
 			mi.return_val.type=Variant::ARRAY;
 			return mi;
 		} break;
+		case GEN_XRANGE: {
+
+			MethodInfo mi("xrange",PropertyInfo(Variant::NIL,"..."));
+			mi.return_val.type=Variant::OBJECT;
+			return mi;
+		} break;
 		case RESOURCE_LOAD: {
 
 			MethodInfo mi("load",PropertyInfo(Variant::STRING,"path"));

+ 1 - 0
modules/gdscript/gd_functions.h

@@ -92,6 +92,7 @@ public:
 		VAR_TO_BYTES,
 		BYTES_TO_VAR,
 		GEN_RANGE,
+		GEN_XRANGE,
 		RESOURCE_LOAD,
 		INST2DICT,
 		DICT2INST,

+ 14 - 0
modules/gdscript/gd_parser.cpp

@@ -1779,6 +1779,20 @@ void GDParser::_parse_block(BlockNode *p_block,bool p_static) {
 					return;
 				}
 
+				// Little optimisation for common usage "for i in range(...):":
+				// don't create and initialize a possibly huge array as range()
+				// would do, but instead create an iterator using xrange()
+				if (container->type == Node::TYPE_OPERATOR) {
+					OperatorNode *op = static_cast<OperatorNode *>(container);
+					if (op->arguments.size() > 0 &&
+							op->arguments[0]->type == Node::TYPE_BUILT_IN_FUNCTION) {
+						BuiltInFunctionNode *c = static_cast<BuiltInFunctionNode *>(op->arguments[0]);
+						if (c->function == GDFunctions::GEN_RANGE) {
+							c->function = GDFunctions::GEN_XRANGE;
+						}
+					}
+				}
+
 				ControlFlowNode *cf_for = alloc_node<ControlFlowNode>();
 
 				cf_for->cf_type=ControlFlowNode::CF_FOR;