svn merge -r 31211:31313 https://svn.blender.org/svnroot/bf-blender/trunk/blender
This commit is contained in:
		| @@ -15,7 +15,7 @@ | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License | ||||
| # along with this program; if not, write to the Free Software Foundation, | ||||
| # Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA. | ||||
| # Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
| # | ||||
| # The Original Code is Copyright (C) Blender Foundation. | ||||
| # All rights reserved. | ||||
|   | ||||
| @@ -45,6 +45,7 @@ import rna_info | ||||
| reload(rna_info) | ||||
|  | ||||
| # lame, python wont give some access | ||||
| ClassMethodDescriptorType = type(dict.__dict__['fromkeys']) | ||||
| MethodDescriptorType = type(dict.get) | ||||
| GetSetDescriptorType = type(int.real) | ||||
|  | ||||
| @@ -153,10 +154,10 @@ def py_descr2sphinx(ident, fw, descr, module_name, type_name, identifier): | ||||
|     if type(descr) == GetSetDescriptorType: | ||||
|         fw(ident + ".. attribute:: %s\n\n" % identifier) | ||||
|         write_indented_lines(ident + "   ", fw, doc, False) | ||||
|     elif type(descr) == MethodDescriptorType: # GetSetDescriptorType's are not documented yet | ||||
|     elif type(descr) in (MethodDescriptorType, ClassMethodDescriptorType): | ||||
|         write_indented_lines(ident, fw, doc, False) | ||||
|     else: | ||||
|         raise TypeError("type was not GetSetDescriptorType or MethodDescriptorType") | ||||
|         raise TypeError("type was not GetSetDescriptorType, MethodDescriptorType or ClassMethodDescriptorType") | ||||
|  | ||||
|     write_example_ref(ident, fw, module_name + "." + type_name + "." + identifier) | ||||
|     fw("\n") | ||||
| @@ -270,7 +271,11 @@ def pymodule2sphinx(BASEPATH, module_name, module, title): | ||||
|         descr_items = [(key, descr) for key, descr in sorted(value.__dict__.items()) if not key.startswith("__")] | ||||
|  | ||||
|         for key, descr in descr_items: | ||||
|             if type(descr) == MethodDescriptorType: # GetSetDescriptorType's are not documented yet | ||||
|             if type(descr) == ClassMethodDescriptorType: | ||||
|                 py_descr2sphinx("   ", fw, descr, module_name, type_name, key) | ||||
|  | ||||
|         for key, descr in descr_items: | ||||
|             if type(descr) == MethodDescriptorType: | ||||
|                 py_descr2sphinx("   ", fw, descr, module_name, type_name, key) | ||||
|  | ||||
|         for key, descr in descr_items: | ||||
|   | ||||
| @@ -15,7 +15,7 @@ | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License | ||||
| # along with this program; if not, write to the Free Software Foundation, | ||||
| # Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA. | ||||
| # Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
| # | ||||
| # The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | ||||
| # All rights reserved. | ||||
|   | ||||
| @@ -46,6 +46,13 @@ | ||||
|  * - Vector.toTrackQuat --> Vector.to_track_quat | ||||
|  * - Quaternion * Quaternion --> cross product (not dot product) | ||||
|  * | ||||
|  * moved into class functions. | ||||
|  * - Mathutils.RotationMatrix -> mathutils.Matrix.Rotation | ||||
|  * - Mathutils.ScaleMatrix -> mathutils.Matrix.Scale | ||||
|  * - Mathutils.ShearMatrix -> mathutils.Matrix.Shear | ||||
|  * - Mathutils.TranslationMatrix -> mathutils.Matrix.Translation | ||||
|  * - Mathutils.OrthoProjectionMatrix -> mathutils.Matrix.OrthoProjection | ||||
|  * | ||||
|  * Moved to Geometry module: Intersect, TriangleArea, TriangleNormal, QuadNormal, LineIntersect | ||||
|  */ | ||||
|  | ||||
| @@ -94,434 +101,7 @@ int mathutils_array_parse(float *array, int array_min, int array_max, PyObject * | ||||
| } | ||||
|  | ||||
| //----------------------------------MATRIX FUNCTIONS-------------------- | ||||
| //----------------------------------mathutils.RotationMatrix() ---------- | ||||
| //mat is a 1D array of floats - row[0][0],row[0][1], row[1][0], etc. | ||||
| static char M_Mathutils_RotationMatrix_doc[] = | ||||
| ".. function:: RotationMatrix(angle, size, axis)\n" | ||||
| "\n" | ||||
| "   Create a matrix representing a rotation.\n" | ||||
| "\n" | ||||
| "   :arg angle: The angle of rotation desired, in radians.\n" | ||||
| "   :type angle: float\n" | ||||
| "   :arg size: The size of the rotation matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :arg axis: a string in ['X', 'Y', 'Z'] or a 3D Vector Object (optional when size is 2).\n" | ||||
| "   :type axis: string or :class:`Vector`\n" | ||||
| "   :return: A new rotation matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *M_Mathutils_RotationMatrix(PyObject * self, PyObject * args) | ||||
| { | ||||
| 	VectorObject *vec= NULL; | ||||
| 	char *axis= NULL; | ||||
| 	int matSize; | ||||
| 	float angle = 0.0f; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
|  | ||||
| 	if(!PyArg_ParseTuple(args, "fi|O", &angle, &matSize, &vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.RotationMatrix(angle, size, axis): expected float int and a string or vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	if(vec && !VectorObject_Check(vec)) { | ||||
| 		axis= _PyUnicode_AsString((PyObject *)vec); | ||||
| 		if(axis==NULL || axis[0]=='\0' || axis[1]!='\0' || axis[0] < 'X' || axis[0] > 'Z') { | ||||
| 			PyErr_SetString(PyExc_TypeError, "mathutils.RotationMatrix(): 3rd argument axis value must be a 3D vector or a string in 'X', 'Y', 'Z'\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		else { | ||||
| 			/* use the string */ | ||||
| 			vec= NULL; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	while (angle<-(Py_PI*2)) | ||||
| 		angle+=(Py_PI*2); | ||||
| 	while (angle>(Py_PI*2)) | ||||
| 		angle-=(Py_PI*2); | ||||
| 	 | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize == 2 && (vec != NULL)) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): cannot create a 2x2 rotation matrix around arbitrary axis\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if((matSize == 3 || matSize == 4) && (axis == NULL) && (vec == NULL)) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): please choose an axis of rotation for 3d and 4d matrices\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec) { | ||||
| 		if(vec->size != 3) { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): the vector axis must be a 3D vector\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		 | ||||
| 		if(!BaseMath_ReadCallback(vec)) | ||||
| 			return NULL; | ||||
| 		 | ||||
| 	} | ||||
|  | ||||
| 	/* check for valid vector/axis above */ | ||||
| 	if(vec) { | ||||
| 		axis_angle_to_mat3( (float (*)[3])mat,vec->vec, angle); | ||||
| 	} | ||||
| 	else if(matSize == 2) { | ||||
| 		//2D rotation matrix | ||||
| 		mat[0] = (float) cos (angle); | ||||
| 		mat[1] = (float) sin (angle); | ||||
| 		mat[2] = -((float) sin(angle)); | ||||
| 		mat[3] = (float) cos(angle); | ||||
| 	} else if(strcmp(axis, "X") == 0) { | ||||
| 		//rotation around X | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[4] = (float) cos(angle); | ||||
| 		mat[5] = (float) sin(angle); | ||||
| 		mat[7] = -((float) sin(angle)); | ||||
| 		mat[8] = (float) cos(angle); | ||||
| 	} else if(strcmp(axis, "Y") == 0) { | ||||
| 		//rotation around Y | ||||
| 		mat[0] = (float) cos(angle); | ||||
| 		mat[2] = -((float) sin(angle)); | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[6] = (float) sin(angle); | ||||
| 		mat[8] = (float) cos(angle); | ||||
| 	} else if(strcmp(axis, "Z") == 0) { | ||||
| 		//rotation around Z | ||||
| 		mat[0] = (float) cos(angle); | ||||
| 		mat[1] = (float) sin(angle); | ||||
| 		mat[3] = -((float) sin(angle)); | ||||
| 		mat[4] = (float) cos(angle); | ||||
| 		mat[8] = 1.0f; | ||||
| 	} | ||||
| 	else { | ||||
| 		/* should never get here */ | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): unknown error\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, NULL); | ||||
| } | ||||
|  | ||||
| static char M_Mathutils_TranslationMatrix_doc[] = | ||||
| ".. function:: TranslationMatrix(vector)\n" | ||||
| "\n" | ||||
| "   Create a matrix representing a translation.\n" | ||||
| "\n" | ||||
| "   :arg vector: The translation vector.\n" | ||||
| "   :type vector: :class:`Vector`\n" | ||||
| "   :return: An identity matrix with a translation.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *M_Mathutils_TranslationMatrix(PyObject * self, VectorObject * vec) | ||||
| { | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
| 	 | ||||
| 	if(!VectorObject_Check(vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.TranslationMatrix(): expected vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec->size != 3 && vec->size != 4) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.TranslationMatrix(): vector must be 3D or 4D\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	 | ||||
| 	if(!BaseMath_ReadCallback(vec)) | ||||
| 		return NULL; | ||||
| 	 | ||||
| 	//create a identity matrix and add translation | ||||
| 	unit_m4((float(*)[4]) mat); | ||||
| 	mat[12] = vec->vec[0]; | ||||
| 	mat[13] = vec->vec[1]; | ||||
| 	mat[14] = vec->vec[2]; | ||||
|  | ||||
| 	return newMatrixObject(mat, 4, 4, Py_NEW, NULL); | ||||
| } | ||||
| //----------------------------------mathutils.ScaleMatrix() ------------- | ||||
| //mat is a 1D array of floats - row[0][0],row[0][1], row[1][0], etc. | ||||
| static char M_Mathutils_ScaleMatrix_doc[] = | ||||
| ".. function:: ScaleMatrix(factor, size, axis)\n" | ||||
| "\n" | ||||
| "   Create a matrix representing a scaling.\n" | ||||
| "\n" | ||||
| "   :arg factor: The factor of scaling to apply.\n" | ||||
| "   :type factor: float\n" | ||||
| "   :arg size: The size of the scale matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :arg axis: Direction to influence scale. (optional).\n" | ||||
| "   :type axis: :class:`Vector`\n" | ||||
| "   :return: A new scale matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *M_Mathutils_ScaleMatrix(PyObject * self, PyObject * args) | ||||
| { | ||||
| 	VectorObject *vec = NULL; | ||||
| 	float norm = 0.0f, factor; | ||||
| 	int matSize, x; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
|  | ||||
| 	if(!PyArg_ParseTuple(args, "fi|O!", &factor, &matSize, &vector_Type, &vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.ScaleMatrix(): expected float int and optional vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.ScaleMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec) { | ||||
| 		if(vec->size > 2 && matSize == 2) { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.ScaleMatrix(): please use 2D vectors when scaling in 2D\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		 | ||||
| 		if(!BaseMath_ReadCallback(vec)) | ||||
| 			return NULL; | ||||
| 		 | ||||
| 	} | ||||
| 	if(vec == NULL) {	//scaling along axis | ||||
| 		if(matSize == 2) { | ||||
| 			mat[0] = factor; | ||||
| 			mat[3] = factor; | ||||
| 		} else { | ||||
| 			mat[0] = factor; | ||||
| 			mat[4] = factor; | ||||
| 			mat[8] = factor; | ||||
| 		} | ||||
| 	} else { //scaling in arbitrary direction | ||||
| 		//normalize arbitrary axis | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			norm += vec->vec[x] * vec->vec[x]; | ||||
| 		} | ||||
| 		norm = (float) sqrt(norm); | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			vec->vec[x] /= norm; | ||||
| 		} | ||||
| 		if(matSize == 2) { | ||||
| 			mat[0] = 1 +((factor - 1) *(vec->vec[0] * vec->vec[0])); | ||||
| 			mat[1] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[2] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[3] = 1 + ((factor - 1) *(vec->vec[1] * vec->vec[1])); | ||||
| 		} else { | ||||
| 			mat[0] = 1 + ((factor - 1) *(vec->vec[0] * vec->vec[0])); | ||||
| 			mat[1] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[2] =((factor - 1) *(vec->vec[0] * vec->vec[2])); | ||||
| 			mat[3] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[4] = 1 + ((factor - 1) *(vec->vec[1] * vec->vec[1])); | ||||
| 			mat[5] =((factor - 1) *(vec->vec[1] * vec->vec[2])); | ||||
| 			mat[6] =((factor - 1) *(vec->vec[0] * vec->vec[2])); | ||||
| 			mat[7] =((factor - 1) *(vec->vec[1] * vec->vec[2])); | ||||
| 			mat[8] = 1 + ((factor - 1) *(vec->vec[2] * vec->vec[2])); | ||||
| 		} | ||||
| 	} | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, NULL); | ||||
| } | ||||
| //----------------------------------mathutils.OrthoProjectionMatrix() --- | ||||
| //mat is a 1D array of floats - row[0][0],row[0][1], row[1][0], etc. | ||||
| static char M_Mathutils_OrthoProjectionMatrix_doc[] = | ||||
| ".. function:: OrthoProjectionMatrix(plane, size, axis)\n" | ||||
| "\n" | ||||
| "   Create a matrix to represent an orthographic projection.\n" | ||||
| "\n" | ||||
| "   :arg plane: Can be any of the following: ['X', 'Y', 'XY', 'XZ', 'YZ', 'R'], where a single axis is for a 2D matrix and 'R' requires axis is given.\n" | ||||
| "   :type plane: string\n" | ||||
| "   :arg size: The size of the projection matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :arg axis: Arbitrary perpendicular plane vector (optional).\n" | ||||
| "   :type axis: :class:`Vector`\n" | ||||
| "   :return: A new projection matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
| static PyObject *M_Mathutils_OrthoProjectionMatrix(PyObject * self, PyObject * args) | ||||
| { | ||||
| 	VectorObject *vec = NULL; | ||||
| 	char *plane; | ||||
| 	int matSize, x; | ||||
| 	float norm = 0.0f; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
| 	 | ||||
| 	if(!PyArg_ParseTuple(args, "si|O!", &plane, &matSize, &vector_Type, &vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.OrthoProjectionMatrix(): expected string and int and optional vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError,"mathutils.OrthoProjectionMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec) { | ||||
| 		if(vec->size > 2 && matSize == 2) { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.OrthoProjectionMatrix(): please use 2D vectors when scaling in 2D\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		 | ||||
| 		if(!BaseMath_ReadCallback(vec)) | ||||
| 			return NULL; | ||||
| 		 | ||||
| 	} | ||||
| 	if(vec == NULL) {	//ortho projection onto cardinal plane | ||||
| 		if((strcmp(plane, "X") == 0) && matSize == 2) { | ||||
| 			mat[0] = 1.0f; | ||||
| 		} else if((strcmp(plane, "Y") == 0) && matSize == 2) { | ||||
| 			mat[3] = 1.0f; | ||||
| 		} else if((strcmp(plane, "XY") == 0) && matSize > 2) { | ||||
| 			mat[0] = 1.0f; | ||||
| 			mat[4] = 1.0f; | ||||
| 		} else if((strcmp(plane, "XZ") == 0) && matSize > 2) { | ||||
| 			mat[0] = 1.0f; | ||||
| 			mat[8] = 1.0f; | ||||
| 		} else if((strcmp(plane, "YZ") == 0) && matSize > 2) { | ||||
| 			mat[4] = 1.0f; | ||||
| 			mat[8] = 1.0f; | ||||
| 		} else { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.OrthoProjectionMatrix(): unknown plane - expected: X, Y, XY, XZ, YZ\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 	} else { //arbitrary plane | ||||
| 		//normalize arbitrary axis | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			norm += vec->vec[x] * vec->vec[x]; | ||||
| 		} | ||||
| 		norm = (float) sqrt(norm); | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			vec->vec[x] /= norm; | ||||
| 		} | ||||
| 		if((strcmp(plane, "R") == 0) && matSize == 2) { | ||||
| 			mat[0] = 1 - (vec->vec[0] * vec->vec[0]); | ||||
| 			mat[1] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[2] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[3] = 1 - (vec->vec[1] * vec->vec[1]); | ||||
| 		} else if((strcmp(plane, "R") == 0) && matSize > 2) { | ||||
| 			mat[0] = 1 - (vec->vec[0] * vec->vec[0]); | ||||
| 			mat[1] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[2] = -(vec->vec[0] * vec->vec[2]); | ||||
| 			mat[3] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[4] = 1 - (vec->vec[1] * vec->vec[1]); | ||||
| 			mat[5] = -(vec->vec[1] * vec->vec[2]); | ||||
| 			mat[6] = -(vec->vec[0] * vec->vec[2]); | ||||
| 			mat[7] = -(vec->vec[1] * vec->vec[2]); | ||||
| 			mat[8] = 1 - (vec->vec[2] * vec->vec[2]); | ||||
| 		} else { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.OrthoProjectionMatrix(): unknown plane - expected: 'r' expected for axis designation\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 	} | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, NULL); | ||||
| } | ||||
|  | ||||
| static char M_Mathutils_ShearMatrix_doc[] = | ||||
| ".. function:: ShearMatrix(plane, factor, size)\n" | ||||
| "\n" | ||||
| "   Create a matrix to represent an shear transformation.\n" | ||||
| "\n" | ||||
| "   :arg plane: Can be any of the following: ['X', 'Y', 'XY', 'XZ', 'YZ'], where a single axis is for a 2D matrix.\n" | ||||
| "   :type plane: string\n" | ||||
| "   :arg factor: The factor of shear to apply.\n" | ||||
| "   :type factor: float\n" | ||||
| "   :arg size: The size of the shear matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :return: A new shear matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *M_Mathutils_ShearMatrix(PyObject * self, PyObject * args) | ||||
| { | ||||
| 	int matSize; | ||||
| 	char *plane; | ||||
| 	float factor; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
|  | ||||
| 	if(!PyArg_ParseTuple(args, "sfi", &plane, &factor, &matSize)) { | ||||
| 		PyErr_SetString(PyExc_TypeError,"mathutils.ShearMatrix(): expected string float and int\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError,"mathutils.ShearMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	if((strcmp(plane, "X") == 0) | ||||
| 		&& matSize == 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[2] = factor; | ||||
| 		mat[3] = 1.0f; | ||||
| 	} else if((strcmp(plane, "Y") == 0) && matSize == 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[1] = factor; | ||||
| 		mat[3] = 1.0f; | ||||
| 	} else if((strcmp(plane, "XY") == 0) && matSize > 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[6] = factor; | ||||
| 		mat[7] = factor; | ||||
| 	} else if((strcmp(plane, "XZ") == 0) && matSize > 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[3] = factor; | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[5] = factor; | ||||
| 		mat[8] = 1.0f; | ||||
| 	} else if((strcmp(plane, "YZ") == 0) && matSize > 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[1] = factor; | ||||
| 		mat[2] = factor; | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[8] = 1.0f; | ||||
| 	} else { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.ShearMatrix(): expected: x, y, xy, xz, yz or wrong matrix size for shearing plane\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, NULL); | ||||
| } | ||||
|  | ||||
| /* Utility functions */ | ||||
|  | ||||
| @@ -647,11 +227,6 @@ void BaseMathObject_dealloc(BaseMathObject * self) | ||||
|  | ||||
| /*----------------------------MODULE INIT-------------------------*/ | ||||
| struct PyMethodDef M_Mathutils_methods[] = { | ||||
| 	{"RotationMatrix", (PyCFunction) M_Mathutils_RotationMatrix, METH_VARARGS, M_Mathutils_RotationMatrix_doc}, | ||||
| 	{"ScaleMatrix", (PyCFunction) M_Mathutils_ScaleMatrix, METH_VARARGS, M_Mathutils_ScaleMatrix_doc}, | ||||
| 	{"ShearMatrix", (PyCFunction) M_Mathutils_ShearMatrix, METH_VARARGS, M_Mathutils_ShearMatrix_doc}, | ||||
| 	{"TranslationMatrix", (PyCFunction) M_Mathutils_TranslationMatrix, METH_O, M_Mathutils_TranslationMatrix_doc}, | ||||
| 	{"OrthoProjectionMatrix", (PyCFunction) M_Mathutils_OrthoProjectionMatrix,  METH_VARARGS, M_Mathutils_OrthoProjectionMatrix_doc}, | ||||
| 	{NULL, NULL, 0, NULL} | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -181,6 +181,438 @@ static PyObject *Matrix_new(PyTypeObject *type, PyObject *args, PyObject *kwds) | ||||
| 	return newMatrixObject(matrix, argSize, seqSize, Py_NEW, NULL); | ||||
| } | ||||
|  | ||||
| /*-----------------------CLASS-METHODS----------------------------*/ | ||||
|  | ||||
| //----------------------------------mathutils.RotationMatrix() ---------- | ||||
| //mat is a 1D array of floats - row[0][0],row[0][1], row[1][0], etc. | ||||
| static char C_Matrix_Rotation_doc[] = | ||||
| ".. classmethod:: Rotation(angle, size, axis)\n" | ||||
| "\n" | ||||
| "   Create a matrix representing a rotation.\n" | ||||
| "\n" | ||||
| "   :arg angle: The angle of rotation desired, in radians.\n" | ||||
| "   :type angle: float\n" | ||||
| "   :arg size: The size of the rotation matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :arg axis: a string in ['X', 'Y', 'Z'] or a 3D Vector Object (optional when size is 2).\n" | ||||
| "   :type axis: string or :class:`Vector`\n" | ||||
| "   :return: A new rotation matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *C_Matrix_Rotation(PyObject *cls, PyObject *args) | ||||
| { | ||||
| 	VectorObject *vec= NULL; | ||||
| 	char *axis= NULL; | ||||
| 	int matSize; | ||||
| 	float angle = 0.0f; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
|  | ||||
| 	if(!PyArg_ParseTuple(args, "fi|O", &angle, &matSize, &vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.RotationMatrix(angle, size, axis): expected float int and a string or vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	if(vec && !VectorObject_Check(vec)) { | ||||
| 		axis= _PyUnicode_AsString((PyObject *)vec); | ||||
| 		if(axis==NULL || axis[0]=='\0' || axis[1]!='\0' || axis[0] < 'X' || axis[0] > 'Z') { | ||||
| 			PyErr_SetString(PyExc_TypeError, "mathutils.RotationMatrix(): 3rd argument axis value must be a 3D vector or a string in 'X', 'Y', 'Z'\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		else { | ||||
| 			/* use the string */ | ||||
| 			vec= NULL; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	while (angle<-(Py_PI*2)) | ||||
| 		angle+=(Py_PI*2); | ||||
| 	while (angle>(Py_PI*2)) | ||||
| 		angle-=(Py_PI*2); | ||||
| 	 | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize == 2 && (vec != NULL)) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): cannot create a 2x2 rotation matrix around arbitrary axis\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if((matSize == 3 || matSize == 4) && (axis == NULL) && (vec == NULL)) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): please choose an axis of rotation for 3d and 4d matrices\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec) { | ||||
| 		if(vec->size != 3) { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): the vector axis must be a 3D vector\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		 | ||||
| 		if(!BaseMath_ReadCallback(vec)) | ||||
| 			return NULL; | ||||
| 		 | ||||
| 	} | ||||
|  | ||||
| 	/* check for valid vector/axis above */ | ||||
| 	if(vec) { | ||||
| 		axis_angle_to_mat3( (float (*)[3])mat,vec->vec, angle); | ||||
| 	} | ||||
| 	else if(matSize == 2) { | ||||
| 		//2D rotation matrix | ||||
| 		mat[0] = (float) cos (angle); | ||||
| 		mat[1] = (float) sin (angle); | ||||
| 		mat[2] = -((float) sin(angle)); | ||||
| 		mat[3] = (float) cos(angle); | ||||
| 	} else if(strcmp(axis, "X") == 0) { | ||||
| 		//rotation around X | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[4] = (float) cos(angle); | ||||
| 		mat[5] = (float) sin(angle); | ||||
| 		mat[7] = -((float) sin(angle)); | ||||
| 		mat[8] = (float) cos(angle); | ||||
| 	} else if(strcmp(axis, "Y") == 0) { | ||||
| 		//rotation around Y | ||||
| 		mat[0] = (float) cos(angle); | ||||
| 		mat[2] = -((float) sin(angle)); | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[6] = (float) sin(angle); | ||||
| 		mat[8] = (float) cos(angle); | ||||
| 	} else if(strcmp(axis, "Z") == 0) { | ||||
| 		//rotation around Z | ||||
| 		mat[0] = (float) cos(angle); | ||||
| 		mat[1] = (float) sin(angle); | ||||
| 		mat[3] = -((float) sin(angle)); | ||||
| 		mat[4] = (float) cos(angle); | ||||
| 		mat[8] = 1.0f; | ||||
| 	} | ||||
| 	else { | ||||
| 		/* should never get here */ | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.RotationMatrix(): unknown error\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, (PyTypeObject *)cls); | ||||
| } | ||||
|  | ||||
|  | ||||
| static char C_Matrix_Translation_doc[] = | ||||
| ".. classmethod:: Translation(vector)\n" | ||||
| "\n" | ||||
| "   Create a matrix representing a translation.\n" | ||||
| "\n" | ||||
| "   :arg vector: The translation vector.\n" | ||||
| "   :type vector: :class:`Vector`\n" | ||||
| "   :return: An identity matrix with a translation.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *C_Matrix_Translation(PyObject *cls, VectorObject * vec) | ||||
| { | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
| 	 | ||||
| 	if(!VectorObject_Check(vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.TranslationMatrix(): expected vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec->size != 3 && vec->size != 4) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.TranslationMatrix(): vector must be 3D or 4D\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	 | ||||
| 	if(!BaseMath_ReadCallback(vec)) | ||||
| 		return NULL; | ||||
| 	 | ||||
| 	//create a identity matrix and add translation | ||||
| 	unit_m4((float(*)[4]) mat); | ||||
| 	mat[12] = vec->vec[0]; | ||||
| 	mat[13] = vec->vec[1]; | ||||
| 	mat[14] = vec->vec[2]; | ||||
|  | ||||
| 	return newMatrixObject(mat, 4, 4, Py_NEW, (PyTypeObject *)cls); | ||||
| } | ||||
| //----------------------------------mathutils.ScaleMatrix() ------------- | ||||
| //mat is a 1D array of floats - row[0][0],row[0][1], row[1][0], etc. | ||||
| static char C_Matrix_Scale_doc[] = | ||||
| ".. classmethod:: Scale(factor, size, axis)\n" | ||||
| "\n" | ||||
| "   Create a matrix representing a scaling.\n" | ||||
| "\n" | ||||
| "   :arg factor: The factor of scaling to apply.\n" | ||||
| "   :type factor: float\n" | ||||
| "   :arg size: The size of the scale matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :arg axis: Direction to influence scale. (optional).\n" | ||||
| "   :type axis: :class:`Vector`\n" | ||||
| "   :return: A new scale matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *C_Matrix_Scale(PyObject *cls, PyObject *args) | ||||
| { | ||||
| 	VectorObject *vec = NULL; | ||||
| 	float norm = 0.0f, factor; | ||||
| 	int matSize, x; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
|  | ||||
| 	if(!PyArg_ParseTuple(args, "fi|O!", &factor, &matSize, &vector_Type, &vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.ScaleMatrix(): expected float int and optional vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.ScaleMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec) { | ||||
| 		if(vec->size > 2 && matSize == 2) { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.ScaleMatrix(): please use 2D vectors when scaling in 2D\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		 | ||||
| 		if(!BaseMath_ReadCallback(vec)) | ||||
| 			return NULL; | ||||
| 		 | ||||
| 	} | ||||
| 	if(vec == NULL) {	//scaling along axis | ||||
| 		if(matSize == 2) { | ||||
| 			mat[0] = factor; | ||||
| 			mat[3] = factor; | ||||
| 		} else { | ||||
| 			mat[0] = factor; | ||||
| 			mat[4] = factor; | ||||
| 			mat[8] = factor; | ||||
| 		} | ||||
| 	} else { //scaling in arbitrary direction | ||||
| 		//normalize arbitrary axis | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			norm += vec->vec[x] * vec->vec[x]; | ||||
| 		} | ||||
| 		norm = (float) sqrt(norm); | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			vec->vec[x] /= norm; | ||||
| 		} | ||||
| 		if(matSize == 2) { | ||||
| 			mat[0] = 1 +((factor - 1) *(vec->vec[0] * vec->vec[0])); | ||||
| 			mat[1] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[2] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[3] = 1 + ((factor - 1) *(vec->vec[1] * vec->vec[1])); | ||||
| 		} else { | ||||
| 			mat[0] = 1 + ((factor - 1) *(vec->vec[0] * vec->vec[0])); | ||||
| 			mat[1] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[2] =((factor - 1) *(vec->vec[0] * vec->vec[2])); | ||||
| 			mat[3] =((factor - 1) *(vec->vec[0] * vec->vec[1])); | ||||
| 			mat[4] = 1 + ((factor - 1) *(vec->vec[1] * vec->vec[1])); | ||||
| 			mat[5] =((factor - 1) *(vec->vec[1] * vec->vec[2])); | ||||
| 			mat[6] =((factor - 1) *(vec->vec[0] * vec->vec[2])); | ||||
| 			mat[7] =((factor - 1) *(vec->vec[1] * vec->vec[2])); | ||||
| 			mat[8] = 1 + ((factor - 1) *(vec->vec[2] * vec->vec[2])); | ||||
| 		} | ||||
| 	} | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, (PyTypeObject *)cls); | ||||
| } | ||||
| //----------------------------------mathutils.OrthoProjectionMatrix() --- | ||||
| //mat is a 1D array of floats - row[0][0],row[0][1], row[1][0], etc. | ||||
| static char C_Matrix_OrthoProjection_doc[] = | ||||
| ".. classmethod:: OrthoProjection(plane, size, axis)\n" | ||||
| "\n" | ||||
| "   Create a matrix to represent an orthographic projection.\n" | ||||
| "\n" | ||||
| "   :arg plane: Can be any of the following: ['X', 'Y', 'XY', 'XZ', 'YZ', 'R'], where a single axis is for a 2D matrix and 'R' requires axis is given.\n" | ||||
| "   :type plane: string\n" | ||||
| "   :arg size: The size of the projection matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :arg axis: Arbitrary perpendicular plane vector (optional).\n" | ||||
| "   :type axis: :class:`Vector`\n" | ||||
| "   :return: A new projection matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
| static PyObject *C_Matrix_OrthoProjection(PyObject *cls, PyObject *args) | ||||
| { | ||||
| 	VectorObject *vec = NULL; | ||||
| 	char *plane; | ||||
| 	int matSize, x; | ||||
| 	float norm = 0.0f; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
| 	 | ||||
| 	if(!PyArg_ParseTuple(args, "si|O!", &plane, &matSize, &vector_Type, &vec)) { | ||||
| 		PyErr_SetString(PyExc_TypeError, "mathutils.OrthoProjectionMatrix(): expected string and int and optional vector\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError,"mathutils.OrthoProjectionMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(vec) { | ||||
| 		if(vec->size > 2 && matSize == 2) { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.OrthoProjectionMatrix(): please use 2D vectors when scaling in 2D\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 		 | ||||
| 		if(!BaseMath_ReadCallback(vec)) | ||||
| 			return NULL; | ||||
| 		 | ||||
| 	} | ||||
| 	if(vec == NULL) {	//ortho projection onto cardinal plane | ||||
| 		if((strcmp(plane, "X") == 0) && matSize == 2) { | ||||
| 			mat[0] = 1.0f; | ||||
| 		} else if((strcmp(plane, "Y") == 0) && matSize == 2) { | ||||
| 			mat[3] = 1.0f; | ||||
| 		} else if((strcmp(plane, "XY") == 0) && matSize > 2) { | ||||
| 			mat[0] = 1.0f; | ||||
| 			mat[4] = 1.0f; | ||||
| 		} else if((strcmp(plane, "XZ") == 0) && matSize > 2) { | ||||
| 			mat[0] = 1.0f; | ||||
| 			mat[8] = 1.0f; | ||||
| 		} else if((strcmp(plane, "YZ") == 0) && matSize > 2) { | ||||
| 			mat[4] = 1.0f; | ||||
| 			mat[8] = 1.0f; | ||||
| 		} else { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.OrthoProjectionMatrix(): unknown plane - expected: X, Y, XY, XZ, YZ\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 	} else { //arbitrary plane | ||||
| 		//normalize arbitrary axis | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			norm += vec->vec[x] * vec->vec[x]; | ||||
| 		} | ||||
| 		norm = (float) sqrt(norm); | ||||
| 		for(x = 0; x < vec->size; x++) { | ||||
| 			vec->vec[x] /= norm; | ||||
| 		} | ||||
| 		if((strcmp(plane, "R") == 0) && matSize == 2) { | ||||
| 			mat[0] = 1 - (vec->vec[0] * vec->vec[0]); | ||||
| 			mat[1] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[2] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[3] = 1 - (vec->vec[1] * vec->vec[1]); | ||||
| 		} else if((strcmp(plane, "R") == 0) && matSize > 2) { | ||||
| 			mat[0] = 1 - (vec->vec[0] * vec->vec[0]); | ||||
| 			mat[1] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[2] = -(vec->vec[0] * vec->vec[2]); | ||||
| 			mat[3] = -(vec->vec[0] * vec->vec[1]); | ||||
| 			mat[4] = 1 - (vec->vec[1] * vec->vec[1]); | ||||
| 			mat[5] = -(vec->vec[1] * vec->vec[2]); | ||||
| 			mat[6] = -(vec->vec[0] * vec->vec[2]); | ||||
| 			mat[7] = -(vec->vec[1] * vec->vec[2]); | ||||
| 			mat[8] = 1 - (vec->vec[2] * vec->vec[2]); | ||||
| 		} else { | ||||
| 			PyErr_SetString(PyExc_AttributeError, "mathutils.OrthoProjectionMatrix(): unknown plane - expected: 'r' expected for axis designation\n"); | ||||
| 			return NULL; | ||||
| 		} | ||||
| 	} | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, (PyTypeObject *)cls); | ||||
| } | ||||
|  | ||||
| static char C_Matrix_Shear_doc[] = | ||||
| ".. classmethod:: Shear(plane, factor, size)\n" | ||||
| "\n" | ||||
| "   Create a matrix to represent an shear transformation.\n" | ||||
| "\n" | ||||
| "   :arg plane: Can be any of the following: ['X', 'Y', 'XY', 'XZ', 'YZ'], where a single axis is for a 2D matrix.\n" | ||||
| "   :type plane: string\n" | ||||
| "   :arg factor: The factor of shear to apply.\n" | ||||
| "   :type factor: float\n" | ||||
| "   :arg size: The size of the shear matrix to construct [2, 4].\n" | ||||
| "   :type size: int\n" | ||||
| "   :return: A new shear matrix.\n" | ||||
| "   :rtype: :class:`Matrix`\n"; | ||||
|  | ||||
| static PyObject *C_Matrix_Shear(PyObject *cls, PyObject *args) | ||||
| { | ||||
| 	int matSize; | ||||
| 	char *plane; | ||||
| 	float factor; | ||||
| 	float mat[16] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, | ||||
| 		0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}; | ||||
|  | ||||
| 	if(!PyArg_ParseTuple(args, "sfi", &plane, &factor, &matSize)) { | ||||
| 		PyErr_SetString(PyExc_TypeError,"mathutils.ShearMatrix(): expected string float and int\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize != 2 && matSize != 3 && matSize != 4) { | ||||
| 		PyErr_SetString(PyExc_AttributeError,"mathutils.ShearMatrix(): can only return a 2x2 3x3 or 4x4 matrix\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	if((strcmp(plane, "X") == 0) | ||||
| 		&& matSize == 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[2] = factor; | ||||
| 		mat[3] = 1.0f; | ||||
| 	} else if((strcmp(plane, "Y") == 0) && matSize == 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[1] = factor; | ||||
| 		mat[3] = 1.0f; | ||||
| 	} else if((strcmp(plane, "XY") == 0) && matSize > 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[6] = factor; | ||||
| 		mat[7] = factor; | ||||
| 	} else if((strcmp(plane, "XZ") == 0) && matSize > 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[3] = factor; | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[5] = factor; | ||||
| 		mat[8] = 1.0f; | ||||
| 	} else if((strcmp(plane, "YZ") == 0) && matSize > 2) { | ||||
| 		mat[0] = 1.0f; | ||||
| 		mat[1] = factor; | ||||
| 		mat[2] = factor; | ||||
| 		mat[4] = 1.0f; | ||||
| 		mat[8] = 1.0f; | ||||
| 	} else { | ||||
| 		PyErr_SetString(PyExc_AttributeError, "mathutils.ShearMatrix(): expected: x, y, xy, xz, yz or wrong matrix size for shearing plane\n"); | ||||
| 		return NULL; | ||||
| 	} | ||||
| 	if(matSize == 4) { | ||||
| 		//resize matrix | ||||
| 		mat[10] = mat[8]; | ||||
| 		mat[9] = mat[7]; | ||||
| 		mat[8] = mat[6]; | ||||
| 		mat[7] = 0.0f; | ||||
| 		mat[6] = mat[5]; | ||||
| 		mat[5] = mat[4]; | ||||
| 		mat[4] = mat[3]; | ||||
| 		mat[3] = 0.0f; | ||||
| 	} | ||||
| 	//pass to matrix creation | ||||
| 	return newMatrixObject(mat, matSize, matSize, Py_NEW, (PyTypeObject *)cls); | ||||
| } | ||||
|  | ||||
| /* assumes rowsize == colsize is checked and the read callback has run */ | ||||
| static float matrix_determinant(MatrixObject * self) | ||||
| { | ||||
| @@ -1326,6 +1758,13 @@ static struct PyMethodDef Matrix_methods[] = { | ||||
| 	{"to_quat", (PyCFunction) Matrix_toQuat, METH_NOARGS, Matrix_toQuat_doc}, | ||||
| 	{"copy", (PyCFunction) Matrix_copy, METH_NOARGS, Matrix_copy_doc}, | ||||
| 	{"__copy__", (PyCFunction) Matrix_copy, METH_NOARGS, Matrix_copy_doc}, | ||||
| 	 | ||||
| 	/* class methods */ | ||||
| 	{"Rotation", (PyCFunction) C_Matrix_Rotation, METH_VARARGS | METH_CLASS, C_Matrix_Rotation_doc}, | ||||
| 	{"Scale", (PyCFunction) C_Matrix_Scale, METH_VARARGS | METH_CLASS, C_Matrix_Scale_doc}, | ||||
| 	{"Shear", (PyCFunction) C_Matrix_Shear, METH_VARARGS | METH_CLASS, C_Matrix_Shear_doc}, | ||||
| 	{"Translation", (PyCFunction) C_Matrix_Translation, METH_O | METH_CLASS, C_Matrix_Translation_doc}, | ||||
| 	{"OrthoProjection", (PyCFunction) C_Matrix_OrthoProjection,  METH_VARARGS | METH_CLASS, C_Matrix_OrthoProjection_doc}, | ||||
| 	{NULL, NULL, 0, NULL} | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -18,7 +18,7 @@ | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software Foundation, | ||||
|  * Inc., 59 Temple Place - Suite 330, Boston, MA	02111-1307, USA. | ||||
|  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
|  * | ||||
|  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | ||||
|  * All rights reserved. | ||||
|   | ||||
| @@ -15,7 +15,7 @@ | ||||
| # | ||||
| # You should have received a copy of the GNU General Public License | ||||
| # along with this program; if not, write to the Free Software Foundation, | ||||
| # Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA. | ||||
| # Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | ||||
| # | ||||
| # The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | ||||
| # All rights reserved. | ||||
|   | ||||
| @@ -35,6 +35,7 @@ | ||||
|  | ||||
| #include "MEM_guardedalloc.h" | ||||
| #include "BKE_utildefines.h" | ||||
| #include "BKE_idcode.h" | ||||
| #include "BKE_context.h" | ||||
| #include "BKE_global.h" /* evil G.* */ | ||||
| #include "BKE_report.h" | ||||
| @@ -418,25 +419,50 @@ static PyObject *pyrna_prop_richcmp(PyObject *a, PyObject *b, int op) | ||||
| } | ||||
|  | ||||
| /*----------------------repr--------------------------------------------*/ | ||||
| static PyObject *pyrna_struct_repr( BPy_StructRNA *self ) | ||||
| static PyObject *pyrna_struct_str( BPy_StructRNA *self ) | ||||
| { | ||||
| 	PyObject *pyob; | ||||
| 	PyObject *ret; | ||||
| 	char *name; | ||||
|  | ||||
| 	/* print name if available */ | ||||
| 	name= RNA_struct_name_get_alloc(&self->ptr, NULL, FALSE); | ||||
| 	if(name) { | ||||
| 		pyob= PyUnicode_FromFormat( "<bpy_struct, %.200s(\"%.200s\")>", RNA_struct_identifier(self->ptr.type), name); | ||||
| 		ret= PyUnicode_FromFormat( "<bpy_struct, %.200s(\"%.200s\")>", RNA_struct_identifier(self->ptr.type), name); | ||||
| 		MEM_freeN(name); | ||||
| 		return pyob; | ||||
| 		return ret; | ||||
| 	} | ||||
|  | ||||
| 	return PyUnicode_FromFormat( "<bpy_struct, %.200s at %p>", RNA_struct_identifier(self->ptr.type), self->ptr.data); | ||||
| } | ||||
|  | ||||
| static PyObject *pyrna_prop_repr( BPy_PropertyRNA *self ) | ||||
| static PyObject *pyrna_struct_repr(BPy_StructRNA *self) | ||||
| { | ||||
| 	PyObject *pyob; | ||||
| 	ID *id= self->ptr.id.data; | ||||
| 	if(id == NULL) | ||||
| 		return pyrna_struct_str(self); /* fallback */ | ||||
| 	 | ||||
| 	if(RNA_struct_is_ID(self->ptr.type)) { | ||||
| 		return PyUnicode_FromFormat( "bpy.data.%s[\"%s\"]", BKE_idcode_to_name_plural(GS(id->name)), id->name+2); | ||||
| 	} | ||||
| 	else { | ||||
| 		PyObject *ret; | ||||
| 		char *path; | ||||
| 		path= RNA_path_from_ID_to_struct(&self->ptr); | ||||
| 		if(path) { | ||||
| 			ret= PyUnicode_FromFormat( "bpy.data.%s[\"%s\"].%s", BKE_idcode_to_name_plural(GS(id->name)), id->name+2, path); | ||||
| 			MEM_freeN(path); | ||||
| 		} | ||||
| 		else { /* cant find, print something sane */ | ||||
| 			ret= PyUnicode_FromFormat( "bpy.data.%s[\"%s\"]...%s", BKE_idcode_to_name_plural(GS(id->name)), id->name+2, RNA_struct_identifier(self->ptr.type)); | ||||
| 		} | ||||
| 		 | ||||
| 		return ret; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static PyObject *pyrna_prop_str( BPy_PropertyRNA *self ) | ||||
| { | ||||
| 	PyObject *ret; | ||||
| 	PointerRNA ptr; | ||||
| 	char *name; | ||||
| 	const char *type_id= NULL; | ||||
| @@ -470,15 +496,36 @@ static PyObject *pyrna_prop_repr( BPy_PropertyRNA *self ) | ||||
| 		name= RNA_struct_name_get_alloc(&ptr, NULL, FALSE); | ||||
|  | ||||
| 		if(name) { | ||||
| 			pyob= PyUnicode_FromFormat( "<bpy_%.200s, %.200s.%.200s(\"%.200s\")>", type_fmt, RNA_struct_identifier(self->ptr.type), RNA_property_identifier(self->prop), name); | ||||
| 			ret= PyUnicode_FromFormat( "<bpy_%.200s, %.200s.%.200s(\"%.200s\")>", type_fmt, RNA_struct_identifier(self->ptr.type), RNA_property_identifier(self->prop), name); | ||||
| 			MEM_freeN(name); | ||||
| 			return pyob; | ||||
| 			return ret; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return PyUnicode_FromFormat( "<bpy_%.200s, %.200s.%.200s>", type_fmt, RNA_struct_identifier(self->ptr.type), RNA_property_identifier(self->prop)); | ||||
| } | ||||
|  | ||||
| static PyObject *pyrna_prop_repr(BPy_PropertyRNA *self) | ||||
| { | ||||
| 	ID *id= self->ptr.id.data; | ||||
| 	PyObject *ret; | ||||
| 	char *path; | ||||
| 	 | ||||
| 	if(id == NULL) | ||||
| 		return pyrna_prop_str(self); /* fallback */ | ||||
| 	 | ||||
| 	path= RNA_path_from_ID_to_property(&self->ptr, self->prop); | ||||
| 	if(path) { | ||||
| 		ret= PyUnicode_FromFormat( "bpy.data.%s[\"%s\"].%s", BKE_idcode_to_name_plural(GS(id->name)), id->name+2, path); | ||||
| 		MEM_freeN(path); | ||||
| 	} | ||||
| 	else { /* cant find, print something sane */ | ||||
| 		ret= PyUnicode_FromFormat( "bpy.data.%s[\"%s\"]...%s", BKE_idcode_to_name_plural(GS(id->name)), id->name+2, RNA_property_identifier(self->prop)); | ||||
| 	} | ||||
| 	 | ||||
| 	return ret; | ||||
| } | ||||
|  | ||||
| static long pyrna_struct_hash( BPy_StructRNA *self ) | ||||
| { | ||||
| 	return _Py_HashPointer(self->ptr.data); | ||||
| @@ -2039,12 +2086,34 @@ static char pyrna_struct_is_property_set_doc[] = | ||||
|  | ||||
| static PyObject *pyrna_struct_is_property_set(BPy_StructRNA *self, PyObject *args) | ||||
| { | ||||
| 	PropertyRNA *prop; | ||||
| 	char *name; | ||||
| 	int ret; | ||||
|  | ||||
| 	if (!PyArg_ParseTuple(args, "s:is_property_set", &name)) | ||||
| 		return NULL; | ||||
|  | ||||
| 	return PyBool_FromLong(RNA_property_is_set(&self->ptr, name)); | ||||
| 	if((prop= RNA_struct_find_property(&self->ptr, name)) == NULL) { | ||||
| 		PyErr_Format(PyExc_TypeError, "%.200s.is_property_set(\"%.200s\") not found", RNA_struct_identifier(self->ptr.type), name); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	/* double property lookup, could speed up */ | ||||
| 	/* return PyBool_FromLong(RNA_property_is_set(&self->ptr, name)); */ | ||||
| 	if(RNA_property_flag(prop) & PROP_IDPROPERTY) { | ||||
| 		IDProperty *group= RNA_struct_idproperties(&self->ptr, 0);		 | ||||
| 		if(group) { | ||||
| 			ret= IDP_GetPropertyFromGroup(group, name) ? 1:0; | ||||
| 		} | ||||
| 		else { | ||||
| 			ret= 0; | ||||
| 		} | ||||
| 	} | ||||
| 	else { | ||||
| 		ret= 1; | ||||
| 	} | ||||
| 	 | ||||
| 	return PyBool_FromLong(ret); | ||||
| } | ||||
|  | ||||
| static char pyrna_struct_is_property_hidden_doc[] = | ||||
| @@ -2059,15 +2128,16 @@ static PyObject *pyrna_struct_is_property_hidden(BPy_StructRNA *self, PyObject * | ||||
| { | ||||
| 	PropertyRNA *prop; | ||||
| 	char *name; | ||||
| 	int hidden; | ||||
|  | ||||
| 	if (!PyArg_ParseTuple(args, "s:is_property_hidden", &name)) | ||||
| 		return NULL; | ||||
| 	 | ||||
| 	prop= RNA_struct_find_property(&self->ptr, name); | ||||
| 	hidden= (prop)? (RNA_property_flag(prop) & PROP_HIDDEN): 1; | ||||
|  | ||||
| 	return PyBool_FromLong(hidden); | ||||
| 	if((prop= RNA_struct_find_property(&self->ptr, name)) == NULL) { | ||||
| 		PyErr_Format(PyExc_TypeError, "%.200s.is_property_hidden(\"%.200s\") not found", RNA_struct_identifier(self->ptr.type), name); | ||||
| 		return NULL; | ||||
| 	} | ||||
|  | ||||
| 	return PyBool_FromLong(RNA_property_flag(prop) & PROP_HIDDEN); | ||||
| } | ||||
|  | ||||
| static char pyrna_struct_path_resolve_doc[] = | ||||
| @@ -3530,7 +3600,7 @@ PyTypeObject pyrna_struct_Type = { | ||||
|  | ||||
| 	( hashfunc )pyrna_struct_hash,	/* hashfunc tp_hash; */ | ||||
| 	NULL,						/* ternaryfunc tp_call; */ | ||||
| 	NULL,                       /* reprfunc tp_str; */ | ||||
| 	(reprfunc) pyrna_struct_str, /* reprfunc tp_str; */ | ||||
| 	( getattrofunc ) pyrna_struct_getattro,	/* getattrofunc tp_getattro; */ | ||||
| 	( setattrofunc ) pyrna_struct_setattro,	/* setattrofunc tp_setattro; */ | ||||
|  | ||||
| @@ -3597,7 +3667,7 @@ PyTypeObject pyrna_prop_Type = { | ||||
| 	NULL,						/* getattrfunc tp_getattr; */ | ||||
| 	NULL,                       /* setattrfunc tp_setattr; */ | ||||
| 	NULL,						/* tp_compare */ /* DEPRECATED in python 3.0! */ | ||||
| 	( reprfunc ) pyrna_prop_repr,	/* tp_repr */ | ||||
| 	(reprfunc) pyrna_prop_repr,	/* tp_repr */ | ||||
|  | ||||
| 	/* Method suites for standard classes */ | ||||
|  | ||||
| @@ -3609,7 +3679,7 @@ PyTypeObject pyrna_prop_Type = { | ||||
|  | ||||
| 	( hashfunc ) pyrna_prop_hash,	/* hashfunc tp_hash; */ | ||||
| 	NULL,                       /* ternaryfunc tp_call; */ | ||||
| 	NULL,                       /* reprfunc tp_str; */ | ||||
| 	(reprfunc) pyrna_prop_str,  /* reprfunc tp_str; */ | ||||
|  | ||||
| 	/* will only use these if this is a subtype of a py class */ | ||||
| 	NULL,						/* getattrofunc tp_getattro; */ | ||||
|   | ||||
		Reference in New Issue
	
	Block a user