bmesh py api
* add BLI_rfindlink for reverse index lookup (used so bm.select_history[-1] doesn't have to loop the entire list twice). * add bm.select_history.active so you can get the last selected item or None without having to check seq length.
This commit is contained in:
		| @@ -23,10 +23,11 @@ | ||||
|  * ***** END GPL LICENSE BLOCK ***** | ||||
|  */ | ||||
|  | ||||
| /** \file blender/python/bmesh/bmesh_py_api.c | ||||
| /** \file blender/python/bmesh/bmesh_py_select.c | ||||
|  *  \ingroup pybmesh | ||||
|  * | ||||
|  * This file defines the 'bmesh' module. | ||||
|  * This file defines the types for 'BMesh.select_history' | ||||
|  * sequence and iterator. | ||||
|  */ | ||||
|  | ||||
| #include <Python.h> | ||||
| @@ -47,9 +48,24 @@ | ||||
|  | ||||
| #include "bmesh_py_api.h" /* own include */ | ||||
|  | ||||
| static PyGetSetDef bpy_bmeditselseq_getseters[] = { | ||||
|     // {(char *)"verts", (getter)bpy_bmeditselseq_get, (setter)NULL, (char *)bpy_bmesh_verts_doc, (void *)BM_VERTS_OF_MESH}, | ||||
| PyDoc_STRVAR(bpy_bmeditselseq_active_doc, | ||||
| "The last selected element or None (read-only).\n\n:type: :class:`BMVert`, :class:`BMEdge` or :class:`BMFace`" | ||||
| ); | ||||
| static PyObject *bpy_bmeditselseq_active_get(BPy_BMEditSelSeq *self, void *UNUSED(closure)) | ||||
| { | ||||
| 	BMEditSelection *ese; | ||||
| 	BPY_BM_CHECK_OBJ(self); | ||||
|  | ||||
| 	if ((ese = self->bm->selected.last)) { | ||||
| 		return BPy_BMElem_CreatePyObject(self->bm, &ese->ele->head); | ||||
| 	} | ||||
| 	else { | ||||
| 		Py_RETURN_NONE; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static PyGetSetDef bpy_bmeditselseq_getseters[] = { | ||||
|     {(char *)"active", (getter)bpy_bmeditselseq_active_get, (setter)NULL, (char *)bpy_bmeditselseq_active_doc, (void *)BM_VERTS_OF_MESH}, | ||||
|     {NULL, NULL, NULL, NULL, NULL} /* Sentinel */ | ||||
| }; | ||||
|  | ||||
| @@ -71,19 +87,25 @@ static Py_ssize_t bpy_bmeditselseq_length(BPy_BMEditSelSeq *self) | ||||
|  | ||||
| static PyObject *bpy_bmeditselseq_subscript_int(BPy_BMEditSelSeq *self, int keynum) | ||||
| { | ||||
| 	BMEditSelection *ese; | ||||
|  | ||||
| 	BPY_BM_CHECK_OBJ(self); | ||||
|  | ||||
| 	if (keynum < 0) keynum += bpy_bmeditselseq_length(self); /* only get length on negative value, may loop entire seq */ | ||||
| 	if (keynum >= 0) { | ||||
| 		BMEditSelection *ese = BLI_findlink(&self->bm->selected, keynum); | ||||
| 		if (ese) { | ||||
| 			return BPy_BMElem_CreatePyObject(self->bm, &ese->ele->head); | ||||
| 		} | ||||
| 	if (keynum < 0) { | ||||
| 		ese = BLI_rfindlink(&self->bm->selected, -1 - keynum); | ||||
| 	} | ||||
| 	else { | ||||
| 		ese = BLI_findlink(&self->bm->selected, keynum); | ||||
| 	} | ||||
|  | ||||
| 	PyErr_Format(PyExc_IndexError, | ||||
| 	             "BMElemSeq[index]: index %d out of range", keynum); | ||||
| 	return NULL; | ||||
| 	if (ese) { | ||||
| 		return BPy_BMElem_CreatePyObject(self->bm, &ese->ele->head); | ||||
| 	} | ||||
| 	else { | ||||
| 		PyErr_Format(PyExc_IndexError, | ||||
| 			         "BMElemSeq[index]: index %d out of range", keynum); | ||||
| 		return NULL; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static PyObject *bpy_bmeditselseq_subscript_slice(BPy_BMEditSelSeq *self, Py_ssize_t start, Py_ssize_t stop) | ||||
|   | ||||
| @@ -1564,7 +1564,7 @@ static PyObject *bpy_bmedgeseq_get(BPy_BMElemSeq *self, PyObject *args) | ||||
| 			return NULL; | ||||
| 		} | ||||
|  | ||||
| 		if ((e=BM_edge_exists(vert_array[0], vert_array[1]))) { | ||||
| 		if ((e = BM_edge_exists(vert_array[0], vert_array[1]))) { | ||||
| 			ret = BPy_BMEdge_CreatePyObject(bm, e); | ||||
| 		} | ||||
| 		else { | ||||
| @@ -2572,7 +2572,7 @@ void *BPy_BMElem_PySeq_As_Array(BMesh **r_bm, PyObject *seq, Py_ssize_t min, Py_ | ||||
| 	PyObject *seq_fast; | ||||
| 	*r_size = 0; | ||||
|  | ||||
| 	if (!(seq_fast=PySequence_Fast(seq, error_prefix))) { | ||||
| 	if (!(seq_fast = PySequence_Fast(seq, error_prefix))) { | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	else { | ||||
|   | ||||
| @@ -50,11 +50,11 @@ int PyC_AsArray(void *array, PyObject *value, const Py_ssize_t length, | ||||
| 	Py_ssize_t value_len; | ||||
| 	Py_ssize_t i; | ||||
|  | ||||
| 	if (!(value_fast=PySequence_Fast(value, error_prefix))) { | ||||
| 	if (!(value_fast = PySequence_Fast(value, error_prefix))) { | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	value_len= PySequence_Fast_GET_SIZE(value_fast); | ||||
| 	value_len = PySequence_Fast_GET_SIZE(value_fast); | ||||
|  | ||||
| 	if (value_len != length) { | ||||
| 		Py_DECREF(value); | ||||
| @@ -69,13 +69,13 @@ int PyC_AsArray(void *array, PyObject *value, const Py_ssize_t length, | ||||
| 		if (is_double) { | ||||
| 			double *array_double= array; | ||||
| 			for (i=0; i<length; i++) { | ||||
| 				array_double[i]= PyFloat_AsDouble(PySequence_Fast_GET_ITEM(value_fast, i)); | ||||
| 				array_double[i] = PyFloat_AsDouble(PySequence_Fast_GET_ITEM(value_fast, i)); | ||||
| 			} | ||||
| 		} | ||||
| 		else { | ||||
| 			float *array_float= array; | ||||
| 			for (i=0; i<length; i++) { | ||||
| 				array_float[i]= PyFloat_AsDouble(PySequence_Fast_GET_ITEM(value_fast, i)); | ||||
| 				array_float[i] = PyFloat_AsDouble(PySequence_Fast_GET_ITEM(value_fast, i)); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| @@ -83,13 +83,13 @@ int PyC_AsArray(void *array, PyObject *value, const Py_ssize_t length, | ||||
| 		/* could use is_double for 'long int' but no use now */ | ||||
| 		int *array_int= array; | ||||
| 		for (i=0; i<length; i++) { | ||||
| 			array_int[i]= PyLong_AsSsize_t(PySequence_Fast_GET_ITEM(value_fast, i)); | ||||
| 			array_int[i] = PyLong_AsSsize_t(PySequence_Fast_GET_ITEM(value_fast, i)); | ||||
| 		} | ||||
| 	} | ||||
| 	else if (type == &PyBool_Type) { | ||||
| 		int *array_bool= array; | ||||
| 		for (i=0; i<length; i++) { | ||||
| 			array_bool[i]= (PyLong_AsSsize_t(PySequence_Fast_GET_ITEM(value_fast, i)) != 0); | ||||
| 			array_bool[i] = (PyLong_AsSsize_t(PySequence_Fast_GET_ITEM(value_fast, i)) != 0); | ||||
| 		} | ||||
| 	} | ||||
| 	else { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user