- adds a new point class * point/ vector math (p + v = p, p - p = v, etc.) * points can be transformed by matrices/quats * wraps 'place vector' type vectors that have no magnitude - wrapped toXXX() methods work correctly * toXXX() will NOT wrap data (this is due to the fact that wrapped data cannot be converted) * added a 'wrapped' attribute to mathutils classes to determine wether the object is accessing python or blender data - added the ability to negate vectors/points with "-vec" * deprecated vector.negate() - added the ability to shorhand inverse matrices with "~mat" (tilde) - conversion between vector/point with toXXX() methods
		
			
				
	
	
		
			416 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			416 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * $Id$
 | |
|  *
 | |
|  * ***** BEGIN GPL/BL DUAL LICENSE BLOCK *****
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version. The Blender
 | |
|  * Foundation also sells licenses for use in proprietary software under
 | |
|  * the Blender License.  See http://www.blender.org/BL/ for information
 | |
|  * about this.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * 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.
 | |
|  *
 | |
|  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
 | |
|  * All rights reserved.
 | |
|  *
 | |
|  * 
 | |
|  * Contributor(s): Joseph Gilbert
 | |
|  *
 | |
|  * ***** END GPL/BL DUAL LICENSE BLOCK *****
 | |
|  */
 | |
| 
 | |
| #include "Mathutils.h"
 | |
| 
 | |
| #include "BLI_arithb.h"
 | |
| #include "BKE_utildefines.h"
 | |
| #include "BLI_blenlib.h"
 | |
| #include "gen_utils.h"
 | |
| 
 | |
| 
 | |
| //-------------------------DOC STRINGS ---------------------------
 | |
| char Euler_Zero_doc[] = "() - set all values in the euler to 0";
 | |
| char Euler_Unique_doc[] ="() - sets the euler rotation a unique shortest arc rotation - tests for gimbal lock";
 | |
| char Euler_ToMatrix_doc[] =	"() - returns a rotation matrix representing the euler rotation";
 | |
| char Euler_ToQuat_doc[] = "() - returns a quaternion representing the euler rotation";
 | |
| char Euler_Rotate_doc[] = "() - rotate a euler by certain amount around an axis of rotation";
 | |
| //-----------------------METHOD DEFINITIONS ----------------------
 | |
| struct PyMethodDef Euler_methods[] = {
 | |
| 	{"zero", (PyCFunction) Euler_Zero, METH_NOARGS, Euler_Zero_doc},
 | |
| 	{"unique", (PyCFunction) Euler_Unique, METH_NOARGS, Euler_Unique_doc},
 | |
| 	{"toMatrix", (PyCFunction) Euler_ToMatrix, METH_NOARGS, Euler_ToMatrix_doc},
 | |
| 	{"toQuat", (PyCFunction) Euler_ToQuat, METH_NOARGS, Euler_ToQuat_doc},
 | |
| 	{"rotate", (PyCFunction) Euler_Rotate, METH_VARARGS, Euler_Rotate_doc},
 | |
| 	{NULL, NULL, 0, NULL}
 | |
| };
 | |
| //-----------------------------METHODS----------------------------
 | |
| //----------------------------Euler.toQuat()----------------------
 | |
| //return a quaternion representation of the euler
 | |
| PyObject *Euler_ToQuat(EulerObject * self)
 | |
| {
 | |
| 	float eul[3];
 | |
| 	float quat[4];
 | |
| 	int x;
 | |
| 
 | |
| 	for(x = 0; x < 3; x++) {
 | |
| 		eul[x] = self->eul[x] * ((float)Py_PI / 180);
 | |
| 	}
 | |
| 	EulToQuat(eul, quat);
 | |
| 	return (PyObject *) newQuaternionObject(quat, Py_NEW);
 | |
| }
 | |
| //----------------------------Euler.toMatrix()---------------------
 | |
| //return a matrix representation of the euler
 | |
| PyObject *Euler_ToMatrix(EulerObject * self)
 | |
| {
 | |
| 	float eul[3];
 | |
| 	float mat[9] = {0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f};
 | |
| 	int x;
 | |
| 
 | |
| 	for(x = 0; x < 3; x++) {
 | |
| 		eul[x] = self->eul[x] * ((float)Py_PI / 180);
 | |
| 	}
 | |
| 	EulToMat3(eul, (float (*)[3]) mat);
 | |
| 	return (PyObject *) newMatrixObject(mat, 3, 3 , Py_NEW);
 | |
| }
 | |
| //----------------------------Euler.unique()-----------------------
 | |
| //sets the x,y,z values to a unique euler rotation
 | |
| PyObject *Euler_Unique(EulerObject * self)
 | |
| {
 | |
| 	double heading, pitch, bank;
 | |
| 	double pi2 =  Py_PI * 2.0f;
 | |
| 	double piO2 = Py_PI / 2.0f;
 | |
| 	double Opi2 = 1.0f / pi2;
 | |
| 
 | |
| 	//radians
 | |
| 	heading = self->eul[0] * (float)Py_PI / 180;
 | |
| 	pitch = self->eul[1] * (float)Py_PI / 180;
 | |
| 	bank = self->eul[2] * (float)Py_PI / 180;
 | |
| 
 | |
| 	//wrap heading in +180 / -180
 | |
| 	pitch += Py_PI;
 | |
| 	pitch -= floor(pitch * Opi2) * pi2;
 | |
| 	pitch -= Py_PI;
 | |
| 
 | |
| 
 | |
| 	if(pitch < -piO2) {
 | |
| 		pitch = -Py_PI - pitch;
 | |
| 		heading += Py_PI;
 | |
| 		bank += Py_PI;
 | |
| 	} else if(pitch > piO2) {
 | |
| 		pitch = Py_PI - pitch;
 | |
| 		heading += Py_PI;
 | |
| 		bank += Py_PI;
 | |
| 	}
 | |
| 	//gimbal lock test
 | |
| 	if(fabs(pitch) > piO2 - 1e-4) {
 | |
| 		heading += bank;
 | |
| 		bank = 0.0f;
 | |
| 	} else {
 | |
| 		bank += Py_PI;
 | |
| 		bank -= (floor(bank * Opi2)) * pi2;
 | |
| 		bank -= Py_PI;
 | |
| 	}
 | |
| 
 | |
| 	heading += Py_PI;
 | |
| 	heading -= (floor(heading * Opi2)) * pi2;
 | |
| 	heading -= Py_PI;
 | |
| 
 | |
| 	//back to degrees
 | |
| 	self->eul[0] = (float)(heading * 180 / (float)Py_PI);
 | |
| 	self->eul[1] = (float)(pitch * 180 / (float)Py_PI);
 | |
| 	self->eul[2] = (float)(bank * 180 / (float)Py_PI);
 | |
| 
 | |
| 	return EXPP_incr_ret((PyObject*)self);
 | |
| }
 | |
| //----------------------------Euler.zero()-------------------------
 | |
| //sets the euler to 0,0,0
 | |
| PyObject *Euler_Zero(EulerObject * self)
 | |
| {
 | |
| 	self->eul[0] = 0.0;
 | |
| 	self->eul[1] = 0.0;
 | |
| 	self->eul[2] = 0.0;
 | |
| 
 | |
| 	return EXPP_incr_ret((PyObject*)self);
 | |
| }
 | |
| //----------------------------Euler.rotate()-----------------------
 | |
| //rotates a euler a certain amount and returns the result
 | |
| //should return a unique euler rotation (i.e. no 720 degree pitches :)
 | |
| PyObject *Euler_Rotate(EulerObject * self, PyObject *args)
 | |
| {
 | |
| 	float angle = 0.0f;
 | |
| 	char *axis;
 | |
| 	int x;
 | |
| 
 | |
| 	if(!PyArg_ParseTuple(args, "fs", &angle, &axis)){
 | |
| 		return EXPP_ReturnPyObjError(PyExc_TypeError,
 | |
| 			"euler.rotate():expected angle (float) and axis (x,y,z)");
 | |
| 	}
 | |
| 	if(!STREQ3(axis,"x","y","z")){
 | |
| 		return EXPP_ReturnPyObjError(PyExc_TypeError,
 | |
| 			"euler.rotate(): expected axis to be 'x', 'y' or 'z'");
 | |
| 	}
 | |
| 
 | |
| 	//covert to radians
 | |
| 	angle *= ((float)Py_PI / 180);
 | |
| 	for(x = 0; x < 3; x++) {
 | |
| 		self->eul[x] *= ((float)Py_PI / 180);
 | |
| 	}
 | |
| 	euler_rot(self->eul, angle, *axis);
 | |
| 	//convert back from radians
 | |
| 	for(x = 0; x < 3; x++) {
 | |
| 		self->eul[x] *= (180 / (float)Py_PI);
 | |
| 	}
 | |
| 
 | |
| 	return EXPP_incr_ret((PyObject*)self);
 | |
| }
 | |
| //----------------------------dealloc()(internal) ------------------
 | |
| //free the py_object
 | |
| static void Euler_dealloc(EulerObject * self)
 | |
| {
 | |
| 	//only free py_data
 | |
| 	if(self->data.py_data){
 | |
| 		PyMem_Free(self->data.py_data);
 | |
| 	}
 | |
| 	PyObject_DEL(self);
 | |
| }
 | |
| //----------------------------getattr()(internal) ------------------
 | |
| //object.attribute access (get)
 | |
| static PyObject *Euler_getattr(EulerObject * self, char *name)
 | |
| {
 | |
| 	if(STREQ(name,"x")){
 | |
| 		return PyFloat_FromDouble(self->eul[0]);
 | |
| 	}else if(STREQ(name, "y")){
 | |
| 		return PyFloat_FromDouble(self->eul[1]);
 | |
| 	}else if(STREQ(name, "z")){
 | |
| 		return PyFloat_FromDouble(self->eul[2]);
 | |
| 	}
 | |
| 	if(STREQ(name, "wrapped")){
 | |
| 		if(self->wrapped == Py_WRAP)
 | |
| 			return EXPP_incr_ret((PyObject *)Py_True);
 | |
| 		else 
 | |
| 			return EXPP_incr_ret((PyObject *)Py_False);
 | |
| 	}
 | |
| 	return Py_FindMethod(Euler_methods, (PyObject *) self, name);
 | |
| }
 | |
| //----------------------------setattr()(internal) ------------------
 | |
| //object.attribute access (set)
 | |
| static int Euler_setattr(EulerObject * self, char *name, PyObject * e)
 | |
| {
 | |
| 	PyObject *f = NULL;
 | |
| 
 | |
| 	f = PyNumber_Float(e);
 | |
| 	if(f == NULL) { // parsed item not a number
 | |
| 		return EXPP_ReturnIntError(PyExc_TypeError, 
 | |
| 			"euler.attribute = x: argument not a number\n");
 | |
| 	}
 | |
| 
 | |
| 	if(STREQ(name,"x")){
 | |
| 		self->eul[0] = (float)PyFloat_AS_DOUBLE(f);
 | |
| 	}else if(STREQ(name, "y")){
 | |
| 		self->eul[1] = (float)PyFloat_AS_DOUBLE(f);
 | |
| 	}else if(STREQ(name, "z")){
 | |
| 		self->eul[2] = (float)PyFloat_AS_DOUBLE(f);
 | |
| 	}else{
 | |
| 		Py_DECREF(f);
 | |
| 		return EXPP_ReturnIntError(PyExc_AttributeError,
 | |
| 				"euler.attribute = x: unknown attribute\n");
 | |
| 	}
 | |
| 
 | |
| 	Py_DECREF(f);
 | |
| 	return 0;
 | |
| }
 | |
| //----------------------------print object (internal)--------------
 | |
| //print the object to screen
 | |
| static PyObject *Euler_repr(EulerObject * self)
 | |
| {
 | |
| 	int i;
 | |
| 	char buffer[48], str[1024];
 | |
| 
 | |
| 	BLI_strncpy(str,"[",1024);
 | |
| 	for(i = 0; i < 3; i++){
 | |
| 		if(i < (2)){
 | |
| 			sprintf(buffer, "%.6f, ", self->eul[i]);
 | |
| 			strcat(str,buffer);
 | |
| 		}else{
 | |
| 			sprintf(buffer, "%.6f", self->eul[i]);
 | |
| 			strcat(str,buffer);
 | |
| 		}
 | |
| 	}
 | |
| 	strcat(str, "](euler)");
 | |
| 
 | |
| 	return EXPP_incr_ret(PyString_FromString(str));
 | |
| }
 | |
| //---------------------SEQUENCE PROTOCOLS------------------------
 | |
| //----------------------------len(object)------------------------
 | |
| //sequence length
 | |
| static int Euler_len(EulerObject * self)
 | |
| {
 | |
| 	return 3;
 | |
| }
 | |
| //----------------------------object[]---------------------------
 | |
| //sequence accessor (get)
 | |
| static PyObject *Euler_item(EulerObject * self, int i)
 | |
| {
 | |
| 	if(i < 0 || i >= 3)
 | |
| 		return EXPP_ReturnPyObjError(PyExc_IndexError,
 | |
| 		"euler[attribute]: array index out of range\n");
 | |
| 
 | |
| 	return Py_BuildValue("f", self->eul[i]);
 | |
| 
 | |
| }
 | |
| //----------------------------object[]-------------------------
 | |
| //sequence accessor (set)
 | |
| static int Euler_ass_item(EulerObject * self, int i, PyObject * ob)
 | |
| {
 | |
| 	PyObject *f = NULL;
 | |
| 
 | |
| 	f = PyNumber_Float(ob);
 | |
| 	if(f == NULL) { // parsed item not a number
 | |
| 		return EXPP_ReturnIntError(PyExc_TypeError, 
 | |
| 			"euler[attribute] = x: argument not a number\n");
 | |
| 	}
 | |
| 
 | |
| 	if(i < 0 || i >= 3){
 | |
| 		Py_DECREF(f);
 | |
| 		return EXPP_ReturnIntError(PyExc_IndexError,
 | |
| 			"euler[attribute] = x: array assignment index out of range\n");
 | |
| 	}
 | |
| 	self->eul[i] = (float)PyFloat_AS_DOUBLE(f);
 | |
| 	Py_DECREF(f);
 | |
| 	return 0;
 | |
| }
 | |
| //----------------------------object[z:y]------------------------
 | |
| //sequence slice (get)
 | |
| static PyObject *Euler_slice(EulerObject * self, int begin, int end)
 | |
| {
 | |
| 	PyObject *list = NULL;
 | |
| 	int count;
 | |
| 
 | |
| 	CLAMP(begin, 0, 3);
 | |
| 	CLAMP(end, 0, 3);
 | |
| 	begin = MIN2(begin,end);
 | |
| 
 | |
| 	list = PyList_New(end - begin);
 | |
| 	for(count = begin; count < end; count++) {
 | |
| 		PyList_SetItem(list, count - begin,
 | |
| 				PyFloat_FromDouble(self->eul[count]));
 | |
| 	}
 | |
| 
 | |
| 	return list;
 | |
| }
 | |
| //----------------------------object[z:y]------------------------
 | |
| //sequence slice (set)
 | |
| static int Euler_ass_slice(EulerObject * self, int begin, int end,
 | |
| 			     PyObject * seq)
 | |
| {
 | |
| 	int i, y, size = 0;
 | |
| 	float eul[3];
 | |
| 
 | |
| 	CLAMP(begin, 0, 3);
 | |
| 	CLAMP(end, 0, 3);
 | |
| 	begin = MIN2(begin,end);
 | |
| 
 | |
| 	size = PySequence_Length(seq);
 | |
| 	if(size != (end - begin)){
 | |
| 		return EXPP_ReturnIntError(PyExc_TypeError,
 | |
| 			"euler[begin:end] = []: size mismatch in slice assignment\n");
 | |
| 	}
 | |
| 
 | |
| 	for (i = 0; i < size; i++) {
 | |
| 		PyObject *e, *f;
 | |
| 
 | |
| 		e = PySequence_GetItem(seq, i);
 | |
| 		if (e == NULL) { // Failed to read sequence
 | |
| 			return EXPP_ReturnIntError(PyExc_RuntimeError, 
 | |
| 				"euler[begin:end] = []: unable to read sequence\n");
 | |
| 		}
 | |
| 		f = PyNumber_Float(e);
 | |
| 		if(f == NULL) { // parsed item not a number
 | |
| 			Py_DECREF(e);
 | |
| 			return EXPP_ReturnIntError(PyExc_TypeError, 
 | |
| 				"euler[begin:end] = []: sequence argument not a number\n");
 | |
| 		}
 | |
| 		eul[i] = (float)PyFloat_AS_DOUBLE(f);
 | |
| 		EXPP_decr2(f,e);
 | |
| 	}
 | |
| 	//parsed well - now set in vector
 | |
| 	for(y = 0; y < 3; y++){
 | |
| 		self->eul[begin + y] = eul[y];
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| //-----------------PROTCOL DECLARATIONS--------------------------
 | |
| static PySequenceMethods Euler_SeqMethods = {
 | |
| 	(inquiry) Euler_len,						/* sq_length */
 | |
| 	(binaryfunc) 0,								/* sq_concat */
 | |
| 	(intargfunc) 0,								/* sq_repeat */
 | |
| 	(intargfunc) Euler_item,					/* sq_item */
 | |
| 	(intintargfunc) Euler_slice,				/* sq_slice */
 | |
| 	(intobjargproc) Euler_ass_item,				/* sq_ass_item */
 | |
| 	(intintobjargproc) Euler_ass_slice,			/* sq_ass_slice */
 | |
| };
 | |
| //------------------PY_OBECT DEFINITION--------------------------
 | |
| PyTypeObject euler_Type = {
 | |
| 	PyObject_HEAD_INIT(NULL) 
 | |
| 	0,											/*ob_size */
 | |
| 	"euler",									/*tp_name */
 | |
| 	sizeof(EulerObject),						/*tp_basicsize */
 | |
| 	0,											/*tp_itemsize */
 | |
| 	(destructor) Euler_dealloc,					/*tp_dealloc */
 | |
| 	(printfunc) 0,								/*tp_print */
 | |
| 	(getattrfunc) Euler_getattr,				/*tp_getattr */
 | |
| 	(setattrfunc) Euler_setattr,				/*tp_setattr */
 | |
| 	0,											/*tp_compare */
 | |
| 	(reprfunc) Euler_repr,						/*tp_repr */
 | |
| 	0,											/*tp_as_number */
 | |
| 	&Euler_SeqMethods,							/*tp_as_sequence */
 | |
| };
 | |
| //------------------------newEulerObject (internal)-------------
 | |
| //creates a new euler object
 | |
| /*pass Py_WRAP - if vector is a WRAPPER for data allocated by BLENDER
 | |
|  (i.e. it was allocated elsewhere by MEM_mallocN())
 | |
|   pass Py_NEW - if vector is not a WRAPPER and managed by PYTHON
 | |
|  (i.e. it must be created here with PyMEM_malloc())*/
 | |
| PyObject *newEulerObject(float *eul, int type)
 | |
| {
 | |
| 	EulerObject *self;
 | |
| 	int x;
 | |
| 
 | |
| 	euler_Type.ob_type = &PyType_Type;
 | |
| 	self = PyObject_NEW(EulerObject, &euler_Type);
 | |
| 	self->data.blend_data = NULL;
 | |
| 	self->data.py_data = NULL;
 | |
| 
 | |
| 	if(type == Py_WRAP){
 | |
| 		self->data.blend_data = eul;
 | |
| 		self->eul = self->data.blend_data;
 | |
| 		self->wrapped = Py_WRAP;
 | |
| 	}else if (type == Py_NEW){
 | |
| 		self->data.py_data = PyMem_Malloc(3 * sizeof(float));
 | |
| 		self->eul = self->data.py_data;
 | |
| 		if(!eul) { //new empty
 | |
| 			for(x = 0; x < 3; x++) {
 | |
| 				self->eul[x] = 0.0f;
 | |
| 			}
 | |
| 		}else{
 | |
| 			for(x = 0; x < 3; x++){
 | |
| 				self->eul[x] = eul[x];
 | |
| 			}
 | |
| 		}
 | |
| 		self->wrapped = Py_NEW;
 | |
| 	}else{ //bad type
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	return (PyObject *) EXPP_incr_ret((PyObject *)self);
 | |
| }
 | |
| 
 |