199 lines
5.4 KiB
C
199 lines
5.4 KiB
C
/*
|
|
*
|
|
* ***** 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): Willian P. Germano
|
|
*
|
|
* ***** END GPL/BL DUAL LICENSE BLOCK *****
|
|
*/
|
|
|
|
/* This file is the old bpython opy_vector.c with minor modifications */
|
|
|
|
#include "vector.h"
|
|
|
|
/*****************************/
|
|
/* Vector Python Object */
|
|
/*****************************/
|
|
#define VectorObject_Check(v) ((v)->ob_type == &vector_Type)
|
|
|
|
static void Vector_dealloc(VectorObject *self)
|
|
{
|
|
PyObject_DEL (self);
|
|
}
|
|
|
|
static PyObject *Vector_getattr(VectorObject *self, char *name)
|
|
{
|
|
if (self->size==3 && ELEM3(name[0], 'x', 'y', 'z') && name[1]==0)
|
|
return PyFloat_FromDouble(self->vec[ name[0]-'x' ]);
|
|
|
|
return EXPP_ReturnPyObjError(PyExc_AttributeError, "attribute not found");
|
|
}
|
|
|
|
static int Vector_setattr(VectorObject *self, char *name, PyObject *v)
|
|
{
|
|
float val;
|
|
|
|
if (!PyArg_Parse(v, "f", &val))
|
|
return EXPP_ReturnIntError(PyExc_TypeError,
|
|
"expected float argument");
|
|
|
|
if (self->size==3 && ELEM3(name[0], 'x', 'y', 'z') && name[1]==0)
|
|
self->vec[ name[0]-'x' ]= val;
|
|
else
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Vectors Sequence methods */
|
|
|
|
static int Vector_len(VectorObject *self)
|
|
{
|
|
return self->size;
|
|
}
|
|
|
|
static PyObject *Vector_item(VectorObject *self, int i)
|
|
{
|
|
if (i < 0 || i >= self->size)
|
|
return EXPP_ReturnPyObjError (PyExc_IndexError,
|
|
"array index out of range");
|
|
|
|
return Py_BuildValue("f", self->vec[i]);
|
|
}
|
|
|
|
static PyObject *Vector_slice(VectorObject *self, int begin, int end)
|
|
{
|
|
PyObject *list;
|
|
int count;
|
|
|
|
if (begin < 0) begin= 0;
|
|
if (end > self->size) end= self->size;
|
|
if (begin > end) begin= end;
|
|
|
|
list= PyList_New(end-begin);
|
|
|
|
for (count = begin; count < end; count++)
|
|
PyList_SetItem(list, count-begin, PyFloat_FromDouble(self->vec[count]));
|
|
|
|
return list;
|
|
}
|
|
|
|
static int Vector_ass_item(VectorObject *self, int i, PyObject *ob)
|
|
{
|
|
if (i < 0 || i >= self->size)
|
|
return EXPP_ReturnIntError(PyExc_IndexError,
|
|
"array assignment index out of range");
|
|
|
|
if (!PyNumber_Check(ob))
|
|
return EXPP_ReturnIntError(PyExc_IndexError,
|
|
"vector member must be a number");
|
|
|
|
self->vec[i]= PyFloat_AsDouble(ob);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int Vector_ass_slice(VectorObject *self, int begin, int end, PyObject *seq)
|
|
{
|
|
int count;
|
|
|
|
if (begin < 0) begin= 0;
|
|
if (end > self->size) end= self->size;
|
|
if (begin > end) begin= end;
|
|
|
|
if (!PySequence_Check(seq))
|
|
return EXPP_ReturnIntError(PyExc_TypeError,
|
|
"illegal argument type for built-in operation");
|
|
|
|
if (PySequence_Length(seq) != (end - begin))
|
|
return EXPP_ReturnIntError(PyExc_TypeError,
|
|
"size mismatch in slice assignment");
|
|
|
|
for (count = begin; count < end; count++) {
|
|
PyObject *ob = PySequence_GetItem(seq, count);
|
|
|
|
if (!PyArg_Parse(ob, "f", &self->vec[count])) {
|
|
Py_DECREF(ob);
|
|
return -1;
|
|
}
|
|
|
|
Py_DECREF(ob);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static PyObject *Vector_repr (VectorObject *self)
|
|
{
|
|
char buffer[100];
|
|
|
|
sprintf (buffer, "[%.3f, %.3f, %.3f, %.3f]\n",
|
|
self->vec[0], self->vec[1], self->vec[2], self->vec[3]);
|
|
|
|
return PyString_FromString (buffer);
|
|
}
|
|
|
|
static PySequenceMethods Vector_SeqMethods =
|
|
{
|
|
(inquiry) Vector_len, /* sq_length */
|
|
(binaryfunc) 0, /* sq_concat */
|
|
(intargfunc) 0, /* sq_repeat */
|
|
(intargfunc) Vector_item, /* sq_item */
|
|
(intintargfunc) Vector_slice, /* sq_slice */
|
|
(intobjargproc) Vector_ass_item, /* sq_ass_item */
|
|
(intintobjargproc) Vector_ass_slice, /* sq_ass_slice */
|
|
};
|
|
|
|
PyTypeObject vector_Type =
|
|
{
|
|
PyObject_HEAD_INIT(NULL)
|
|
0, /*ob_size*/
|
|
"vector", /*tp_name*/
|
|
sizeof(VectorObject), /*tp_basicsize*/
|
|
0, /*tp_itemsize*/
|
|
/* methods */
|
|
(destructor) Vector_dealloc, /*tp_dealloc*/
|
|
(printfunc) 0, /*tp_print*/
|
|
(getattrfunc) Vector_getattr, /*tp_getattr*/
|
|
(setattrfunc) Vector_setattr, /*tp_setattr*/
|
|
0, /*tp_compare*/
|
|
(reprfunc) Vector_repr, /*tp_repr*/
|
|
0, /*tp_as_number*/
|
|
&Vector_SeqMethods, /*tp_as_sequence*/
|
|
};
|
|
|
|
PyObject *newVectorObject(float *vec, int size)
|
|
{
|
|
VectorObject *self;
|
|
|
|
vector_Type.ob_type = &PyType_Type;
|
|
|
|
self= PyObject_NEW(VectorObject, &vector_Type);
|
|
|
|
self->vec= vec;
|
|
self->size= size;
|
|
|
|
return (PyObject*) self;
|
|
}
|