1646 lines
		
	
	
		
			57 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1646 lines
		
	
	
		
			57 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.
 | |
|  *
 | |
|  * This is a new part of Blender.
 | |
|  *
 | |
|  * Contributor(s): Michel Selten
 | |
|  *
 | |
|  * ***** END GPL/BL DUAL LICENSE BLOCK *****
 | |
| */
 | |
| 
 | |
| #include "Object.h"
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Python API function prototypes for the Blender module.                    */
 | |
| /*****************************************************************************/
 | |
| static PyObject *M_Object_New(PyObject *self, PyObject *args);
 | |
| PyObject *M_Object_Get(PyObject *self, PyObject *args);
 | |
| PyObject *M_Object_get(PyObject *self, PyObject *args);
 | |
| static PyObject *M_Object_GetSelected (PyObject *self, PyObject *args);
 | |
| static PyObject *M_Object_getSelected (PyObject *self, PyObject *args);
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* The following string definitions are used for documentation strings.      */
 | |
| /* In Python these will be written to the console when doing a               */
 | |
| /* Blender.Object.__doc__                                                    */
 | |
| /*****************************************************************************/
 | |
| char M_Object_doc[] =
 | |
| "The Blender Object module\n\n\
 | |
| This module provides access to **Object Data** in Blender.\n";
 | |
| 
 | |
| char M_Object_New_doc[] =
 | |
| "(type) - Add a new object of type 'type' in the current scene";
 | |
| 
 | |
| char M_Object_Get_doc[] =
 | |
| "(name) - return the object with the name 'name', returns None if not\
 | |
|     found.\n\
 | |
|     If 'name' is not specified, it returns a list of all objects in the\n\
 | |
|     current scene.";
 | |
| 
 | |
| char M_Object_GetSelected_doc[] =
 | |
| "() - Returns a list of selected Objects in the active layer(s)\n\
 | |
| The active object is the first in the list, if visible";
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Python method structure definition for Blender.Object module:             */
 | |
| /*****************************************************************************/
 | |
| struct PyMethodDef M_Object_methods[] = {
 | |
|     {"New",         (PyCFunction)M_Object_New,         METH_VARARGS,
 | |
|                     M_Object_New_doc},
 | |
|     {"Get",         (PyCFunction)M_Object_Get,         METH_VARARGS,
 | |
|                     M_Object_Get_doc},
 | |
|     {"get",         (PyCFunction)M_Object_get,         METH_VARARGS,
 | |
|                     M_Object_Get_doc},
 | |
|     {"GetSelected", (PyCFunction)M_Object_GetSelected, METH_VARARGS,
 | |
|                     M_Object_GetSelected_doc},
 | |
|     {"getSelected", (PyCFunction)M_Object_getSelected, METH_VARARGS,
 | |
|                     M_Object_GetSelected_doc},
 | |
|     {NULL, NULL, 0, NULL}
 | |
| };
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Python BPy_Object methods declarations:                                     */
 | |
| /*****************************************************************************/
 | |
| static PyObject *Object_buildParts (BPy_Object *self);
 | |
| static PyObject *Object_clrParent (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_getData (BPy_Object *self);
 | |
| static PyObject *Object_getDeltaLocation (BPy_Object *self);
 | |
| static PyObject *Object_getDrawMode (BPy_Object *self);
 | |
| static PyObject *Object_getDrawType (BPy_Object *self);
 | |
| static PyObject *Object_getEuler (BPy_Object *self);
 | |
| static PyObject *Object_getInverseMatrix (BPy_Object *self);
 | |
| static PyObject *Object_getLocation (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_getMaterials (BPy_Object *self);
 | |
| static PyObject *Object_getMatrix (BPy_Object *self);
 | |
| static PyObject *Object_getName (BPy_Object *self);
 | |
| static PyObject *Object_getParent (BPy_Object *self);
 | |
| static PyObject *Object_getTracked (BPy_Object *self);
 | |
| static PyObject *Object_getType (BPy_Object *self);
 | |
| static PyObject *Object_link (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_makeParent (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_materialUsage (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setDeltaLocation (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setDrawMode (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setDrawType (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setEuler (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setLocation (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setMaterials (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_setName (BPy_Object *self, PyObject *args);
 | |
| static PyObject *Object_shareFrom (BPy_Object *self, PyObject *args);
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Python BPy_Object methods table:                                            */
 | |
| /*****************************************************************************/
 | |
| static PyMethodDef BPy_Object_methods[] = {
 | |
|     /* name, method, flags, doc */
 | |
|     {"buildParts",        (PyCFunction)Object_buildParts,        METH_NOARGS,
 | |
|         "Recalcs particle system (if any) "},  
 | |
| {"clrParent",        (PyCFunction)Object_clrParent,        METH_VARARGS,
 | |
|         "Clears parent object. Optionally specify:\n\
 | |
| mode\n\t2: Keep object transform\nfast\n\t>0: Don't update scene \
 | |
| hierarchy (faster)"},
 | |
|     {"getData",          (PyCFunction)Object_getData,          METH_NOARGS,
 | |
|         "Returns the datablock object containing the object's data, \
 | |
| e.g. Mesh"},
 | |
|     {"getDeltaLocation", (PyCFunction)Object_getDeltaLocation, METH_NOARGS,
 | |
|         "Returns the object's delta location (x, y, z)"},
 | |
|     {"getDrawMode",      (PyCFunction)Object_getDrawMode,      METH_NOARGS,
 | |
|         "Returns the object draw modes"},
 | |
|     {"getDrawType",      (PyCFunction)Object_getDrawType,      METH_NOARGS,
 | |
|         "Returns the object draw type"},
 | |
|     {"getEuler",         (PyCFunction)Object_getEuler,         METH_NOARGS,
 | |
|         "Returns the object's rotation as Euler rotation vector\n\
 | |
| (rotX, rotY, rotZ)"},
 | |
|     {"getInverseMatrix", (PyCFunction)Object_getInverseMatrix, METH_NOARGS,
 | |
|         "Returns the object's inverse matrix"},
 | |
|     {"getLocation",      (PyCFunction)Object_getLocation,      METH_VARARGS,
 | |
|         "Returns the object's location (x, y, z)"},
 | |
|     {"getMaterials",     (PyCFunction)Object_getMaterials,     METH_NOARGS,
 | |
|         "Returns list of materials assigned to the object"},
 | |
|     {"getMatrix",        (PyCFunction)Object_getMatrix,        METH_NOARGS,
 | |
|         "Returns the object matrix"},
 | |
|     {"getName",          (PyCFunction)Object_getName,          METH_NOARGS,
 | |
|         "Returns the name of the object"},
 | |
|     {"getParent",        (PyCFunction)Object_getParent,        METH_NOARGS,
 | |
|         "Returns the object's parent object"},
 | |
|     {"getTracked",       (PyCFunction)Object_getTracked,       METH_NOARGS,
 | |
|         "Returns the object's tracked object"},
 | |
|     {"getType",          (PyCFunction)Object_getType,          METH_NOARGS,
 | |
|         "Returns type of string of Object"},
 | |
|     {"link",             (PyCFunction)Object_link,             METH_VARARGS,
 | |
|         "Links Object with data provided in the argument. The data must \n\
 | |
| match the Object's type, so you cannot link a Lamp to a Mesh type object."},
 | |
|     {"makeParent",       (PyCFunction)Object_makeParent,       METH_VARARGS,
 | |
|         "Makes the object the parent of the objects provided in the \n\
 | |
| argument which must be a list of valid Objects. Optional extra arguments:\n\
 | |
| mode:\n\t0: make parent with inverse\n\t1: without inverse\n\
 | |
| fase:\n\t0: update scene hierarchy automatically\n\t\
 | |
| don't update scene hierarchy (faster). In this case, you must\n\t\
 | |
| explicitely update the Scene hierarchy."},
 | |
|     {"materialUsage",    (PyCFunction)Object_materialUsage,    METH_VARARGS,
 | |
|         "Determines the way the material is used and returs status.\n\
 | |
| Possible arguments (provide as strings):\n\
 | |
| \tData:   Materials assigned to the object's data are shown. (default)\n\
 | |
| \tObject: Materials assigned to the object are shown."},
 | |
|     {"setDeltaLocation", (PyCFunction)Object_setDeltaLocation, METH_VARARGS,
 | |
|         "Sets the object's delta location which must be a vector triple."},
 | |
|     {"setDrawMode",      (PyCFunction)Object_setDrawMode,      METH_VARARGS,
 | |
|         "Sets the object's drawing mode. The argument can be a sum of:\n\
 | |
| 2:  axis\n4:  texspace\n8:  drawname\n16: drawimage\n32: drawwire"},
 | |
|     {"setDrawType",      (PyCFunction)Object_setDrawType,      METH_VARARGS,
 | |
|         "Sets the object's drawing type. The argument must be one of:\n\
 | |
| 1: Bounding box\n2: Wire\n3: Solid\n4: Shaded\n5: Textured"},
 | |
|     {"setEuler",         (PyCFunction)Object_setEuler,         METH_VARARGS,
 | |
|         "Set the object's rotation according to the specified Euler\n\
 | |
| angles. The argument must be a vector triple"},
 | |
|     {"setLocation",      (PyCFunction)Object_setLocation,      METH_VARARGS,
 | |
|         "Set the object's location. The first argument must be a vector\n\
 | |
| triple."},
 | |
|     {"setMaterials",     (PyCFunction)Object_setMaterials,     METH_VARARGS,
 | |
|         "Sets materials. The argument must be a list of valid material\n\
 | |
| objects."},
 | |
|     {"setName",          (PyCFunction)Object_setName,          METH_VARARGS,
 | |
|         "Sets the name of the object"},
 | |
|     {"shareFrom",        (PyCFunction)Object_shareFrom,        METH_VARARGS,
 | |
|         "Link data of self with object specified in the argument. This\n\
 | |
| works only if self and the object specified are of the same type."},
 | |
|     {0}
 | |
| };
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* PythonTypeObject callback function prototypes                             */
 | |
| /*****************************************************************************/
 | |
| static void      Object_dealloc (BPy_Object *obj);
 | |
| static PyObject* Object_getAttr (BPy_Object *obj, char *name);
 | |
| static int       Object_setAttr (BPy_Object *obj, char *name, PyObject *v);
 | |
| static PyObject* Object_repr    (BPy_Object *obj);
 | |
| static int       Object_compare (BPy_Object *a, BPy_Object *b);
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Python TypeObject structure definition.                                   */
 | |
| /*****************************************************************************/
 | |
| PyTypeObject Object_Type =
 | |
| {
 | |
|     PyObject_HEAD_INIT(NULL)
 | |
|     0,                                /* ob_size */
 | |
|     "Blender Object",                 /* tp_name */
 | |
|     sizeof (BPy_Object),              /* tp_basicsize */
 | |
|     0,                                /* tp_itemsize */
 | |
|     /* methods */
 | |
|     (destructor)Object_dealloc,       /* tp_dealloc */
 | |
|     0,                                /* tp_print */
 | |
|     (getattrfunc)Object_getAttr,      /* tp_getattr */
 | |
|     (setattrfunc)Object_setAttr,      /* tp_setattr */
 | |
|     (cmpfunc)Object_compare,          /* tp_compare */
 | |
|     (reprfunc)Object_repr,            /* tp_repr */
 | |
|     0,                                /* tp_as_number */
 | |
|     0,                                /* tp_as_sequence */
 | |
|     0,                                /* tp_as_mapping */
 | |
|     0,                                /* tp_as_hash */
 | |
|     0,0,0,0,0,0,
 | |
|     0,                                /* tp_doc */ 
 | |
|     0,0,0,0,0,0,
 | |
|     BPy_Object_methods,               /* tp_methods */
 | |
|     0,                                /* tp_members */
 | |
| };
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:              M_Object_New                                       */
 | |
| /* Python equivalent:     Blender.Object.New                                 */
 | |
| /*****************************************************************************/
 | |
| PyObject *M_Object_New(PyObject *self, PyObject *args)
 | |
| {
 | |
|     struct Object   * object;
 | |
|     BPy_Object      * blen_object;
 | |
|     int               type;
 | |
|     char            * str_type;
 | |
|     char            * name = NULL;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "s|s", &str_type, &name))
 | |
|     {
 | |
|         PythonReturnErrorObject (PyExc_TypeError,
 | |
|                     "string expected as argument");
 | |
|         return (NULL);
 | |
|     }
 | |
| 
 | |
|     if (strcmp (str_type, "Armature") == 0)     type = OB_ARMATURE;
 | |
|     else if (strcmp (str_type, "Camera") == 0)  type = OB_CAMERA;
 | |
|     else if (strcmp (str_type, "Curve") == 0)   type = OB_CURVE;
 | |
| /*    else if (strcmp (str_type, "Text") == 0)    type = OB_FONT; */
 | |
| /*    else if (strcmp (str_type, "Ika") == 0)     type = OB_IKA; */
 | |
|     else if (strcmp (str_type, "Lamp") == 0)    type = OB_LAMP;
 | |
| /*    else if (strcmp (str_type, "Lattice") == 0) type = OB_LATTICE; */
 | |
| /*    else if (strcmp (str_type, "Mball") == 0)   type = OB_MBALL; */
 | |
|     else if (strcmp (str_type, "Mesh") == 0)    type = OB_MESH;
 | |
| /*    else if (strcmp (str_type, "Surf") == 0)    type = OB_SURF; */
 | |
| /*    else if (strcmp (str_type, "Wave") == 0)    type = OB_WAVE; */
 | |
|     else if (strcmp (str_type, "Empty") == 0)   type = OB_EMPTY;
 | |
|     else
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|             "Unknown type specified"));
 | |
|     }
 | |
| 
 | |
|     /* Create a new object. */
 | |
|     if (name == NULL)
 | |
|     {
 | |
|         /* No name is specified, set the name to the type of the object. */
 | |
|         name = str_type;
 | |
|     }
 | |
|     object = alloc_libblock (&(G.main->object), ID_OB, name);
 | |
| 
 | |
|     object->flag = 0;
 | |
|     object->type = type;
 | |
| 
 | |
|     /* transforms */
 | |
|     QuatOne(object->quat);
 | |
|     QuatOne(object->dquat);
 | |
|     
 | |
|     object->col[3]= 1.0;    // alpha 
 | |
| 
 | |
|     object->size[0] = object->size[1] = object->size[2] = 1.0;
 | |
|     object->loc[0] = object->loc[1] = object->loc[2] = 0.0;
 | |
|     Mat4One(object->parentinv);
 | |
|     Mat4One(object->obmat);
 | |
|     object->dt = OB_SHADED; // drawtype
 | |
| 
 | |
|     if (U.flag & MAT_ON_OB)
 | |
|     {
 | |
|         object->colbits = -1;
 | |
|     }
 | |
|     switch (object->type)
 | |
|     {
 | |
|         case OB_CAMERA: /* fall through. */
 | |
|         case OB_LAMP:
 | |
|             object->trackflag = OB_NEGZ;
 | |
|             object->upflag = OB_POSY;
 | |
|             break;
 | |
|         default:
 | |
|             object->trackflag = OB_POSY;
 | |
|             object->upflag = OB_POSZ;
 | |
|     }
 | |
|     object->ipoflag = OB_OFFS_OB + OB_OFFS_PARENT;
 | |
| 
 | |
|     /* duplivert settings */
 | |
|     object->dupon = 1;
 | |
|     object->dupoff = 0;
 | |
|     object->dupsta = 1;
 | |
|     object->dupend = 100;
 | |
| 
 | |
|     /* Gameengine defaults*/
 | |
|     object->mass = 1.0;
 | |
|     object->inertia = 1.0;
 | |
|     object->formfactor = 0.4;
 | |
|     object->damping = 0.04;
 | |
|     object->rdamping = 0.1;
 | |
|     object->anisotropicFriction[0] = 1.0;
 | |
|     object->anisotropicFriction[1] = 1.0;
 | |
|     object->anisotropicFriction[2] = 1.0;
 | |
|     object->gameflag = OB_PROP;
 | |
|     
 | |
|     object->lay = 1; // Layer, by default visible
 | |
| 
 | |
|     switch(type)
 | |
|     {
 | |
|         case OB_ARMATURE:
 | |
|         /* TODO: Do we need to add something to G? (see the OB_LAMP case) */
 | |
|             object->data = add_armature();
 | |
|             break;
 | |
|         case OB_CAMERA:
 | |
|         /* TODO: Do we need to add something to G? (see the OB_LAMP case) */
 | |
|             object->data = add_camera();
 | |
|             break;
 | |
|         case OB_CURVE:
 | |
|             object->data = add_curve(OB_CURVE);
 | |
|             G.totcurve++;
 | |
|             break;
 | |
|         case OB_LAMP:
 | |
|             object->data = add_lamp();
 | |
|             G.totlamp++;
 | |
|             break;
 | |
|         case OB_MESH:
 | |
|             object->data = add_mesh();
 | |
|             G.totmesh++;
 | |
|             break;
 | |
| 
 | |
|     /* TODO the following types will be supported later
 | |
|         case OB_SURF:
 | |
|             object->data = add_curve(OB_SURF);
 | |
|             G.totcurve++;
 | |
|             break;
 | |
|         case OB_FONT:
 | |
|             object->data = add_curve(OB_FONT);
 | |
|             break;
 | |
|         case OB_MBALL:
 | |
|             object->data = add_mball();
 | |
|             break;
 | |
|         case OB_IKA:
 | |
|             object->data = add_ika();
 | |
|             object->dt = OB_WIRE;
 | |
|             break;
 | |
|         case OB_LATTICE:
 | |
|             object->data = (void *)add_lattice();
 | |
|             object->dt = OB_WIRE;
 | |
|             break;
 | |
|         case OB_WAVE:
 | |
|             object->data = add_wave();
 | |
|             break;
 | |
|     */
 | |
|     }
 | |
| 
 | |
|     G.totobj++;
 | |
| 
 | |
|     /* Create a Python object from it. */
 | |
|     blen_object = (BPy_Object*)PyObject_NEW (BPy_Object, &Object_Type); 
 | |
|     blen_object->object = object;
 | |
|     blen_object->data = NULL;
 | |
|     blen_object->parent = NULL;
 | |
|     blen_object->track = NULL;
 | |
|     blen_object->ipo = NULL;
 | |
| 
 | |
|     return ((PyObject*)blen_object);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:              M_Object_Get                                       */
 | |
| /* Python equivalent:     Blender.Object.Get                                 */
 | |
| /*****************************************************************************/
 | |
| PyObject *M_Object_Get(PyObject *self, PyObject *args)
 | |
| {
 | |
|     struct Object   * object;
 | |
|     BPy_Object      * blen_object;
 | |
|     char            * name = NULL;
 | |
| 
 | |
|     PyArg_ParseTuple(args, "|s", &name);
 | |
| 
 | |
|     if (name != NULL)
 | |
|     {
 | |
|         object = GetObjectByName (name);
 | |
| 
 | |
|         if (object == NULL)
 | |
|         {
 | |
|             /* No object exists with the name specified in the argument name. */
 | |
|             return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                         "Unknown object specified."));
 | |
|         }
 | |
|         blen_object = (BPy_Object*)PyObject_NEW (BPy_Object, &Object_Type); 
 | |
|         blen_object->object = object;
 | |
|         blen_object->parent = NULL;
 | |
|         blen_object->data = NULL;
 | |
|         blen_object->track = NULL;
 | |
|         blen_object->ipo = NULL;
 | |
| 
 | |
|         return ((PyObject*)blen_object);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         /* No argument has been given. Return a list of all objects. */
 | |
|         PyObject    * obj_list;
 | |
|         Link        * link;
 | |
|         int           index;
 | |
| 
 | |
|         obj_list = PyList_New (BLI_countlist (&(G.main->object)));
 | |
| 
 | |
|         if (obj_list == NULL)
 | |
|         {
 | |
|             return (PythonReturnErrorObject (PyExc_SystemError,
 | |
|                     "List creation failed."));
 | |
|         }
 | |
| 
 | |
|         link = G.main->object.first;
 | |
|         index = 0;
 | |
|         while (link)
 | |
|         {
 | |
|             object = (Object*)link;
 | |
|             blen_object = (BPy_Object*)PyObject_NEW (BPy_Object, &Object_Type);
 | |
|             blen_object->object = object;
 | |
|             blen_object->parent = NULL;
 | |
|             blen_object->data = NULL;
 | |
|             blen_object->track = NULL;
 | |
|             blen_object->ipo = NULL;
 | |
| 
 | |
|             PyList_SetItem (obj_list, index, (PyObject*)blen_object);
 | |
|             index++;
 | |
|             link = link->next;
 | |
|         }
 | |
|         return (obj_list);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:              M_Object_get                                       */
 | |
| /* Python equivalent:     Blender.Object.get                                 */
 | |
| /*****************************************************************************/
 | |
| PyObject *M_Object_get(PyObject *self, PyObject *args)
 | |
| {
 | |
|     PyErr_Warn (PyExc_DeprecationWarning,
 | |
|                 "The Object.get() function will be removed in Blender 2.29\n" \
 | |
|                 "Please update the script to use Object.Get");
 | |
|     return (M_Object_Get (self, args));
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:              M_Object_GetSelected                               */
 | |
| /* Python equivalent:     Blender.Object.getSelected                         */
 | |
| /*****************************************************************************/
 | |
| static PyObject *M_Object_GetSelected (PyObject *self, PyObject *args)
 | |
| {
 | |
|     BPy_Object        * blen_object;
 | |
|     PyObject        * list;
 | |
|     Base            * base_iter;
 | |
| 
 | |
|     printf ("In Object_GetSelected()\n");
 | |
| 
 | |
|     list = PyList_New (0);
 | |
|     if ((G.scene->basact) &&
 | |
|         ((G.scene->basact->flag & SELECT) &&
 | |
|          (G.scene->basact->lay & G.vd->lay)))
 | |
|     {
 | |
|         /* Active object is first in the list. */
 | |
|         blen_object = (BPy_Object*)PyObject_NEW (BPy_Object, &Object_Type); 
 | |
|         if (blen_object == NULL)
 | |
|         {
 | |
|             Py_DECREF (list);
 | |
|             Py_INCREF (Py_None);
 | |
|             return (Py_None);
 | |
|         }
 | |
|         blen_object->object = G.scene->basact->object;
 | |
|         blen_object->data = NULL;
 | |
|         blen_object->parent = NULL;
 | |
|         blen_object->track = NULL;
 | |
|         blen_object->ipo = NULL;
 | |
|         PyList_Append (list, (PyObject*)blen_object);
 | |
|     }
 | |
| 
 | |
|     base_iter = G.scene->base.first;
 | |
|     while (base_iter)
 | |
|     {
 | |
|         if (((base_iter->flag & SELECT) &&
 | |
|              (base_iter->lay & G.vd->lay)) &&
 | |
|             (base_iter != G.scene->basact))
 | |
|         {
 | |
|             blen_object = (BPy_Object*)PyObject_NEW (BPy_Object, &Object_Type); 
 | |
|             if (blen_object == NULL)
 | |
|             {
 | |
|                 Py_DECREF (list);
 | |
|                 Py_INCREF (Py_None);
 | |
|                 return (Py_None);
 | |
|             }
 | |
|             blen_object->object = base_iter->object;
 | |
|             blen_object->data = NULL;
 | |
|             blen_object->parent = NULL;
 | |
|             blen_object->track = NULL;
 | |
|             blen_object->ipo = NULL;
 | |
|             PyList_Append (list, (PyObject*)blen_object);
 | |
|         }
 | |
|         base_iter = base_iter->next;
 | |
|     }
 | |
|     return (list);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:              M_Object_getSelected                               */
 | |
| /* Python equivalent:     Blender.Object.getSelected                         */
 | |
| /*****************************************************************************/
 | |
| static PyObject *M_Object_getSelected (PyObject *self, PyObject *args)
 | |
| {
 | |
|     PyErr_Warn (PyExc_DeprecationWarning,
 | |
|                 "The Object.getSelected() function will be removed in "\
 | |
|                 "Blender 2.29\n" \
 | |
|                 "Please update the script to use Object.GetSelected");
 | |
|     return (M_Object_GetSelected (self, args));
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:              initObject                                         */
 | |
| /*****************************************************************************/
 | |
| PyObject *Object_Init (void)
 | |
| {
 | |
|     PyObject    * module;
 | |
| 
 | |
|     Object_Type.ob_type = &PyType_Type;
 | |
| 
 | |
|     module = Py_InitModule3("Blender.Object", M_Object_methods, M_Object_doc);
 | |
| 
 | |
|     return (module);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Python BPy_Object methods:                                                  */
 | |
| /*****************************************************************************/
 | |
| 
 | |
| 
 | |
| static PyObject *Object_buildParts (BPy_Object *self)
 | |
| {
 | |
| void build_particle_system(Object *ob);
 | |
| 	struct Object *obj = self->object;
 | |
| 	build_particle_system( obj);
 | |
|   Py_INCREF (Py_None);
 | |
|             return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_clrParent (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     int       mode=0;
 | |
|     int       fast=0;
 | |
| 
 | |
|     if (!PyArg_ParseTuple (args, "|ii", &mode, &fast))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|             "expected one or two integers as arguments"));
 | |
|     }
 | |
| 
 | |
|     /* Remove the link only, the object is still in the scene. */
 | |
|     self->object->parent = NULL;
 | |
|     self->parent = NULL;
 | |
| 
 | |
|     if (mode == 2)
 | |
|     {
 | |
|         /* Keep transform */
 | |
|         apply_obmat (self->object);
 | |
|     }
 | |
| 
 | |
|     if (!fast)
 | |
|     {
 | |
|         sort_baselist (G.scene);
 | |
|     }
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getData (BPy_Object *self)
 | |
| {
 | |
|     PyObject  * data_object;
 | |
| //#    int         obj_id;
 | |
| //#    ID        * id;
 | |
| 
 | |
|     /* If there's a valid PyObject already, then just return that one. */
 | |
|     if (self->data != NULL)
 | |
|     {
 | |
|         Py_INCREF (self->data);
 | |
|         return (self->data);
 | |
|     }
 | |
| 
 | |
|     /* If there's no data associated to the Object, then there's nothing to */
 | |
|     /* return. */
 | |
|     if (self->object->data == NULL)
 | |
|     {
 | |
|         Py_INCREF (Py_None);
 | |
|         return (Py_None);
 | |
|     }
 | |
| 
 | |
|     data_object = NULL;
 | |
| 
 | |
|     //#id = (ID*)self->object;
 | |
|     //#obj_id = MAKE_ID2 (id->name[0], id->name[1]);
 | |
|     switch (self->object->type)//#obj_id)
 | |
|     {
 | |
|         case OB_ARMATURE://#ID_AR:
 | |
|             data_object = Armature_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case OB_CAMERA://#ID_CA:
 | |
|             data_object = Camera_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case OB_CURVE://#ID_CU:
 | |
|             data_object = Curve_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case ID_IM:
 | |
|             data_object = Image_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case ID_IP:
 | |
|             data_object = Ipo_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case OB_LAMP://#ID_LA:
 | |
|             data_object = Lamp_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case ID_MA:
 | |
|             break;
 | |
|         case OB_MESH://#ID_ME:
 | |
|             data_object = NMesh_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case ID_OB:
 | |
|             data_object = Object_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case ID_SCE:
 | |
|             break;
 | |
|         case ID_TXT:
 | |
|             data_object = Text_CreatePyObject (self->object->data);
 | |
|             break;
 | |
|         case ID_WO:
 | |
|             break;
 | |
|         default:
 | |
|             break;
 | |
|     }
 | |
|     if (data_object == NULL)
 | |
|     {
 | |
|         Py_INCREF (Py_None);
 | |
|         return (Py_None);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         self->data = data_object;
 | |
|         Py_INCREF (data_object);
 | |
|         return (data_object);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getDeltaLocation (BPy_Object *self)
 | |
| {
 | |
|     PyObject *attr = Py_BuildValue ("fff",
 | |
|                                     self->object->dloc[0],
 | |
|                                     self->object->dloc[1],
 | |
|                                     self->object->dloc[2]);
 | |
| 
 | |
|     if (attr) return (attr);
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.dloc attributes"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getDrawMode (BPy_Object *self)
 | |
| {
 | |
|     PyObject *attr = Py_BuildValue ("b", self->object->dtx);
 | |
| 
 | |
|     if (attr) return (attr);
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.drawMode attribute"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getDrawType (BPy_Object *self)
 | |
| {
 | |
|     PyObject *attr = Py_BuildValue ("b", self->object->dt);
 | |
| 
 | |
|     if (attr) return (attr);
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.drawType attribute"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getEuler (BPy_Object *self)
 | |
| {
 | |
|     PyObject *attr = Py_BuildValue ("fff",
 | |
|                                     self->object->drot[0],
 | |
|                                     self->object->drot[1],
 | |
|                                     self->object->drot[2]);
 | |
| 
 | |
|     if (attr) return (attr);
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.drot attributes"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getInverseMatrix (BPy_Object *self)
 | |
| {
 | |
|     MatrixObject *inverse = (MatrixObject *)newMatrixObject (NULL);
 | |
|     
 | |
|     Mat4Invert (inverse->mem, self->object->obmat);
 | |
| 
 | |
|     return ((PyObject *)inverse);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getLocation (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     PyObject *attr = Py_BuildValue ("fff",
 | |
|                                     self->object->loc[0],
 | |
|                                     self->object->loc[1],
 | |
|                                     self->object->loc[2]);
 | |
| 
 | |
|     if (attr) return (attr);
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.loc attributes"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getMaterials (BPy_Object *self)
 | |
| {
 | |
|     return (EXPP_PyList_fromMaterialList (self->object->mat,
 | |
|                                           self->object->totcol));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getMatrix (BPy_Object *self)
 | |
| {
 | |
|     Object  * ob;
 | |
| 
 | |
|     ob = self->object;
 | |
| 
 | |
|     return (newMatrixObject (ob->obmat));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getName (BPy_Object *self)
 | |
| {
 | |
|     PyObject *attr = Py_BuildValue ("s", self->object->id.name+2);
 | |
| 
 | |
|     if (attr) return (attr);
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get the name of the Object"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getParent (BPy_Object *self)
 | |
| {
 | |
|     PyObject *attr;
 | |
| 
 | |
|     if (self->parent)
 | |
|     {
 | |
|         Py_INCREF ((PyObject*)self->parent);
 | |
|         return ((PyObject*)self->parent);
 | |
|     }
 | |
| 
 | |
|     if (self->object->parent == NULL)
 | |
|     {
 | |
|         return (EXPP_incr_ret (Py_None));
 | |
|     }
 | |
|     attr = Object_CreatePyObject (self->object->parent);
 | |
| 
 | |
|     if (attr)
 | |
|     {
 | |
|         self->parent = (struct BPy_Object*)attr;
 | |
|         return (attr);
 | |
|     }
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.parent attribute"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getTracked (BPy_Object *self)
 | |
| {
 | |
|     PyObject    *attr;
 | |
| 
 | |
|     if (self->track)
 | |
|     {
 | |
|         Py_INCREF ((PyObject*)self->track);
 | |
|         return ((PyObject*)self->track);
 | |
|     }
 | |
| 
 | |
|     /* TODO: what if self->object->track==NULL? Should we return Py_None? */
 | |
|     attr = Object_CreatePyObject (self->object->track);
 | |
| 
 | |
|     if (attr)
 | |
|     {
 | |
|         self->track = (struct BPy_Object*)attr;
 | |
|         return (attr);
 | |
|     }
 | |
| 
 | |
|     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|             "couldn't get Object.track attribute"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_getType (BPy_Object *self)
 | |
| {
 | |
|     switch (self->object->type)
 | |
|     {
 | |
|         case OB_ARMATURE:   return (Py_BuildValue ("s", "Armature"));
 | |
|         case OB_CAMERA:     return (Py_BuildValue ("s", "Camera"));
 | |
|         case OB_CURVE:      return (Py_BuildValue ("s", "Curve"));
 | |
|         case OB_EMPTY:      return (Py_BuildValue ("s", "Empty"));
 | |
|         case OB_FONT:       return (Py_BuildValue ("s", "Text"));
 | |
|         case OB_IKA:        return (Py_BuildValue ("s", "Ika"));
 | |
|         case OB_LAMP:       return (Py_BuildValue ("s", "Lamp"));
 | |
|         case OB_LATTICE:    return (Py_BuildValue ("s", "Lattice"));
 | |
|         case OB_MBALL:      return (Py_BuildValue ("s", "MBall"));
 | |
|         case OB_MESH:       return (Py_BuildValue ("s", "Mesh"));
 | |
|         case OB_SURF:       return (Py_BuildValue ("s", "Surf"));
 | |
|         case OB_WAVE:       return (Py_BuildValue ("s", "Wave"));
 | |
|         default:            return (Py_BuildValue ("s", "unknown"));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *Object_link (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     PyObject    * py_data;
 | |
|     ID          * id;
 | |
|     ID          * oldid;
 | |
|     int           obj_id;
 | |
|     void        * data = NULL;
 | |
| 
 | |
|     if (!PyArg_ParseTuple (args, "O", &py_data))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|             "expected an object as argument"));
 | |
|     }
 | |
|     if (Camera_CheckPyObject (py_data))
 | |
|         data = (void *)Camera_FromPyObject (py_data);
 | |
|     if (Lamp_CheckPyObject (py_data))
 | |
|         data = (void *)Lamp_FromPyObject (py_data);
 | |
|     if (Curve_CheckPyObject (py_data))
 | |
|         data = (void *)Curve_FromPyObject (py_data);
 | |
|     /* TODO: add the (N)Mesh check and from functions here when finished. */
 | |
| 
 | |
|     oldid = (ID*) self->object->data;
 | |
|     id = (ID*) data;
 | |
|     obj_id = MAKE_ID2 (id->name[0], id->name[1]);
 | |
| 
 | |
|     switch (obj_id)
 | |
|     {
 | |
|         case ID_CA:
 | |
|             if (self->object->type != OB_CAMERA)
 | |
|             {
 | |
|                 return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "The 'link' object is incompatible with the base object"));
 | |
|             }
 | |
|             break;
 | |
|         case ID_LA:
 | |
|             if (self->object->type != OB_LAMP)
 | |
|             {
 | |
|                 return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "The 'link' object is incompatible with the base object"));
 | |
|             }
 | |
|             break;
 | |
|         case ID_ME:
 | |
|             if (self->object->type != OB_MESH)
 | |
|             {
 | |
|                 return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "The 'link' object is incompatible with the base object"));
 | |
|             }
 | |
|             break;
 | |
|         case ID_CU:
 | |
|             if (self->object->type != OB_CURVE)
 | |
|             {
 | |
|                 return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "The 'link' object is incompatible with the base object"));
 | |
|             }
 | |
|             break;
 | |
|         default:
 | |
|             return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "Linking this object type is not supported"));
 | |
|     }
 | |
|     self->object->data = data;
 | |
|     self->data = py_data;
 | |
|     id_us_plus (id);
 | |
|     if (oldid)
 | |
|     {
 | |
|         if (oldid->us > 0)
 | |
|         {
 | |
|             oldid->us--;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|                 "old object reference count below 0"));
 | |
|         }
 | |
|     }
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_makeParent (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     PyObject    * list;
 | |
|     PyObject    * py_child;
 | |
|     BPy_Object    * py_obj_child;
 | |
|     Object      * child;
 | |
|     Object      * parent;
 | |
|     int           noninverse = 0;
 | |
|     int           fast = 0;
 | |
|     int           i;
 | |
| 
 | |
|     /* Check if the arguments passed to makeParent are valid. */
 | |
|     if (!PyArg_ParseTuple (args, "O|ii", &list, &noninverse, &fast))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|             "expected a list of objects and one or two integers as arguments"));
 | |
|     }
 | |
|     if (!PySequence_Check (list))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_TypeError,
 | |
|             "expected a list of objects"));
 | |
|     }
 | |
| 
 | |
|     /* Check if the PyObject passed in list is a Blender object. */
 | |
|     for (i=0 ; i<PySequence_Length (list) ; i++)
 | |
|     {
 | |
|         child = NULL;
 | |
|         py_child = PySequence_GetItem (list, i);
 | |
|         if (Object_CheckPyObject (py_child))
 | |
|             child = (Object*) Object_FromPyObject (py_child);
 | |
| 
 | |
|         if (child == NULL)
 | |
|         {
 | |
|             return (PythonReturnErrorObject (PyExc_TypeError,
 | |
|                 "Object Type expected"));
 | |
|         }
 | |
| 
 | |
|         parent = (Object*)self->object;
 | |
|         if (test_parent_loop (parent, child))
 | |
|         {
 | |
|             return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|                 "parenting loop detected - parenting failed"));
 | |
|         }
 | |
|         child->partype = PAROBJECT;
 | |
|         child->parent = parent;
 | |
|         py_obj_child = (BPy_Object *) py_child;
 | |
|         py_obj_child->parent = (struct BPy_Object *)self;
 | |
|         if (noninverse == 1)
 | |
|         {
 | |
|             /* Parent inverse = unity */
 | |
|             child->loc[0] = 0.0;
 | |
|             child->loc[1] = 0.0;
 | |
|             child->loc[2] = 0.0;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             what_does_parent (child);
 | |
|             Mat4Invert (child->parentinv, parent->obmat);
 | |
|         }
 | |
| 
 | |
|         if (!fast)
 | |
|         {
 | |
|             sort_baselist (G.scene);
 | |
|         }
 | |
| 
 | |
|         /* We don't need the child object anymore. */
 | |
|         Py_DECREF ((PyObject *) child);
 | |
|     }
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_materialUsage (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     return (PythonReturnErrorObject (PyExc_NotImplementedError,
 | |
|             "materialUsage: not yet implemented"));
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setDeltaLocation (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     float   dloc1;
 | |
|     float   dloc2;
 | |
|     float   dloc3;
 | |
| 
 | |
|     if (!PyArg_ParseTuple (args, "fff", &dloc1, &dloc2, &dloc3))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected three float arguments"));
 | |
|     }
 | |
| 
 | |
|     self->object->dloc[0] = dloc1;
 | |
|     self->object->dloc[1] = dloc2;
 | |
|     self->object->dloc[2] = dloc3;
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setDrawMode (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     char    dt;
 | |
| 
 | |
|     if (!PyArg_Parse (args, "b", &dt))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected an integer as argument"));
 | |
|     }
 | |
|     self->object->dt = dt;
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setDrawType (BPy_Object *self, PyObject *args)
 | |
| { 
 | |
|     char    dtx;
 | |
| 
 | |
|     if (!PyArg_Parse (args, "b", &dtx))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected an integer as argument"));
 | |
|     }
 | |
|     self->object->dtx = dtx;
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setEuler (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     float   drot1;
 | |
|     float   drot2;
 | |
|     float   drot3;
 | |
| 
 | |
|     if (!PyArg_ParseTuple (args, "fff", &drot1, &drot2, &drot3))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected three float arguments"));
 | |
|     }
 | |
| 
 | |
|     self->object->drot[0] = drot1;
 | |
|     self->object->drot[1] = drot2;
 | |
|     self->object->drot[2] = drot3;
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setLocation (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     float   loc1;
 | |
|     float   loc2;
 | |
|     float   loc3;
 | |
| 
 | |
|     if (!PyArg_ParseTuple (args, "fff", &loc1, &loc2, &loc3))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected three float arguments"));
 | |
|     }
 | |
| 
 | |
|     self->object->loc[0] = loc1;
 | |
|     self->object->loc[1] = loc2;
 | |
|     self->object->loc[2] = loc3;
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setMaterials (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     PyObject     * list;
 | |
|     int            len;
 | |
|     int            i;
 | |
|     Material    ** matlist;
 | |
| 
 | |
|     if (!PyArg_Parse (args, "O", &list))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected a list of materials as argument"));
 | |
|     }
 | |
| 
 | |
|     len = PySequence_Length (list);
 | |
|     if (len > 0)
 | |
|     {
 | |
|         matlist = EXPP_newMaterialList_fromPyList (list);
 | |
|         if (!matlist)
 | |
|         {
 | |
|             return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "material list must be a list of valid materials!"));
 | |
|         }
 | |
|         if ((len < 0) || (len > MAXMAT))
 | |
|         {
 | |
|             return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|                 "illegal material index!"));
 | |
|         }
 | |
| 
 | |
|         if (self->object->mat)
 | |
|         {
 | |
|             EXPP_releaseMaterialList (self->object->mat, len);
 | |
|         }
 | |
|         /* Increase the user count on all materials */
 | |
|         for (i=0 ; i<len ; i++)
 | |
|         {
 | |
|             id_us_plus ((ID *) matlist[i]);
 | |
|         }
 | |
|         self->object->mat = matlist;
 | |
|         self->object->totcol = len;
 | |
|         self->object->actcol = -1;
 | |
| 
 | |
|         switch (self->object->type)
 | |
|         {
 | |
|             case OB_CURVE:  /* fall through */
 | |
|             case OB_FONT:   /* fall through */
 | |
|             case OB_MESH:   /* fall through */
 | |
|             case OB_MBALL:  /* fall through */
 | |
|             case OB_SURF:
 | |
|                 EXPP_synchronizeMaterialLists (self->object,
 | |
|                                                self->object->data);
 | |
|                 break;
 | |
|             default:
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_setName (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     char  * name;
 | |
|     char    buf[21];
 | |
| 
 | |
|     if (!PyArg_Parse (args, "s", &name))
 | |
|     {
 | |
|         return (PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected a String as argument"));
 | |
|     }
 | |
| 
 | |
|     PyOS_snprintf(buf, sizeof(buf), "%s", name);
 | |
| 
 | |
|     rename_id(&self->object->id, buf);
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| static PyObject *Object_shareFrom (BPy_Object *self, PyObject *args)
 | |
| {
 | |
|     BPy_Object      * object;
 | |
|     ID              * id;
 | |
|     ID              * oldid;
 | |
| 
 | |
|     if (!PyArg_Parse (args, "O", &object))
 | |
|     {
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                 "expected an object argument");
 | |
|         return (NULL);
 | |
|     }
 | |
|     if (!Object_CheckPyObject ((PyObject*)object))
 | |
|     {
 | |
|         PythonReturnErrorObject (PyExc_TypeError,
 | |
|                 "argument 1 is not of type 'Object'");
 | |
|         return (NULL);
 | |
|     }
 | |
| 
 | |
|     if (self->object->type != object->object->type)
 | |
|     {
 | |
|         PythonReturnErrorObject (PyExc_TypeError,
 | |
|                 "objects are not of same data type");
 | |
|         return (NULL);
 | |
|     }
 | |
|     switch (self->object->type)
 | |
|     {
 | |
|         case OB_MESH:
 | |
|             oldid = (ID*) self->object->data;
 | |
|             id = (ID*) object->data;
 | |
|             self->object->data = object->data;
 | |
|             if (self->data != NULL)
 | |
|             {
 | |
|                 Py_DECREF (self->data);
 | |
|                 self->data = NULL;
 | |
|             }
 | |
|             id_us_plus (id);
 | |
|             if (oldid)
 | |
|             {
 | |
|                 if (oldid->us > 0)
 | |
|                 {
 | |
|                     oldid->us--;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     return (PythonReturnErrorObject (PyExc_RuntimeError,
 | |
|                             "old object reference count below 0"));
 | |
|                 }
 | |
|             }
 | |
|             Py_INCREF (Py_None);
 | |
|             return (Py_None);
 | |
|         default:
 | |
|             PythonReturnErrorObject (PyExc_TypeError,
 | |
|                     "type not supported");
 | |
|             return (NULL);
 | |
|     }
 | |
| 
 | |
|     Py_INCREF (Py_None);
 | |
|     return (Py_None);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_CreatePyObject                                        */
 | |
| /* Description: This function will create a new BlenObject from an existing  */
 | |
| /*              Object structure.                                            */
 | |
| /*****************************************************************************/
 | |
| PyObject* Object_CreatePyObject (struct Object *obj)
 | |
| {
 | |
|     BPy_Object    * blen_object;
 | |
| 
 | |
|     blen_object = (BPy_Object*)PyObject_NEW (BPy_Object, &Object_Type);
 | |
| 
 | |
|     if (blen_object == NULL)
 | |
|     {
 | |
|         return (NULL);
 | |
|     }
 | |
|     blen_object->object = obj;
 | |
|     return ((PyObject*)blen_object);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_CheckPyObject                                         */
 | |
| /* Description: This function returns true when the given PyObject is of the */
 | |
| /*              type Object. Otherwise it will return false.                 */
 | |
| /*****************************************************************************/
 | |
| int Object_CheckPyObject (PyObject *py_obj)
 | |
| {
 | |
|     return (py_obj->ob_type == &Object_Type);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_FromPyObject                                          */
 | |
| /* Description: This function returns the Blender object from the given      */
 | |
| /*              PyObject.                                                    */
 | |
| /*****************************************************************************/
 | |
| struct Object* Object_FromPyObject (PyObject *py_obj)
 | |
| {
 | |
|     BPy_Object    * blen_obj;
 | |
| 
 | |
|     blen_obj = (BPy_Object*)py_obj;
 | |
|     return (blen_obj->object);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Description: Returns the object with the name specified by the argument   */
 | |
| /*              name. Note that the calling function has to remove the first */
 | |
| /*              two characters of the object name. These two characters      */
 | |
| /*              specify the type of the object (OB, ME, WO, ...)             */
 | |
| /*              The function will return NULL when no object with the given  */
 | |
| /*              name is found.                                               */
 | |
| /*****************************************************************************/
 | |
| Object * GetObjectByName (char * name)
 | |
| {
 | |
| 	Object	* obj_iter;
 | |
| 
 | |
| 	obj_iter = G.main->object.first;
 | |
| 	while (obj_iter)
 | |
| 	{
 | |
| 		if (StringEqual (name, GetIdName (&(obj_iter->id))))
 | |
| 		{
 | |
| 			return (obj_iter);
 | |
| 		}
 | |
| 		obj_iter = obj_iter->id.next;
 | |
| 	}
 | |
| 
 | |
| 	/* There is no object with the given name */
 | |
| 	return (NULL);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_dealloc                                               */
 | |
| /* Description: This is a callback function for the BlenObject type. It is   */
 | |
| /*              the destructor function.                                     */
 | |
| /*****************************************************************************/
 | |
| static void Object_dealloc (BPy_Object *obj)
 | |
| {
 | |
|     PyObject_DEL (obj);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_getAttr                                               */
 | |
| /* Description: This is a callback function for the BlenObject type. It is   */
 | |
| /*              the function that retrieves any value from Blender and       */
 | |
| /*              passes it to Python.                                         */
 | |
| /*****************************************************************************/
 | |
| static PyObject* Object_getAttr (BPy_Object *obj, char *name)
 | |
| {
 | |
|     struct Object   * object;
 | |
|     struct Ika      * ika;
 | |
| 
 | |
|     object = obj->object;
 | |
|     if (StringEqual (name, "LocX"))
 | |
|         return (PyFloat_FromDouble(object->loc[0]));
 | |
|     if (StringEqual (name, "LocY"))
 | |
|         return (PyFloat_FromDouble(object->loc[1]));
 | |
|     if (StringEqual (name, "LocZ"))
 | |
|         return (PyFloat_FromDouble(object->loc[2]));
 | |
|     if (StringEqual (name, "loc"))
 | |
|         return (Py_BuildValue ("fff", object->loc[0], object->loc[1],
 | |
|                                object->loc[2]));
 | |
|     if (StringEqual (name, "dLocX"))
 | |
|         return (PyFloat_FromDouble(object->dloc[0]));
 | |
|     if (StringEqual (name, "dLocY"))
 | |
|         return (PyFloat_FromDouble(object->dloc[1]));
 | |
|     if (StringEqual (name, "dLocZ"))
 | |
|         return (PyFloat_FromDouble(object->dloc[2]));
 | |
|     if (StringEqual (name, "dloc"))
 | |
|         return (Py_BuildValue ("fff", object->dloc[0], object->dloc[1],
 | |
|                                object->dloc[2]));
 | |
|     if (StringEqual (name, "RotX"))
 | |
|         return (PyFloat_FromDouble(object->rot[0]));
 | |
|     if (StringEqual (name, "RotY"))
 | |
|         return (PyFloat_FromDouble(object->rot[1]));
 | |
|     if (StringEqual (name, "RotZ"))
 | |
|         return (PyFloat_FromDouble(object->rot[2]));
 | |
|     if (StringEqual (name, "rot"))
 | |
|         return (Py_BuildValue ("fff", object->rot[0], object->rot[1],
 | |
|                                object->rot[2]));
 | |
|     if (StringEqual (name, "dRotX"))
 | |
|         return (PyFloat_FromDouble(object->drot[0]));
 | |
|     if (StringEqual (name, "dRotY"))
 | |
|         return (PyFloat_FromDouble(object->drot[1]));
 | |
|     if (StringEqual (name, "dRotZ"))
 | |
|         return (PyFloat_FromDouble(object->drot[2]));
 | |
|     if (StringEqual (name, "drot"))
 | |
|         return (Py_BuildValue ("fff", object->drot[0], object->drot[1],
 | |
|                                object->drot[2]));
 | |
|     if (StringEqual (name, "SizeX"))
 | |
|         return (PyFloat_FromDouble(object->size[0]));
 | |
|     if (StringEqual (name, "SizeY"))
 | |
|         return (PyFloat_FromDouble(object->size[1]));
 | |
|     if (StringEqual (name, "SizeZ"))
 | |
|         return (PyFloat_FromDouble(object->size[2]));
 | |
|     if (StringEqual (name, "size"))
 | |
|         return (Py_BuildValue ("fff", object->size[0], object->size[1],
 | |
|                                object->size[2]));
 | |
|     if (StringEqual (name, "dSizeX"))
 | |
|         return (PyFloat_FromDouble(object->dsize[0]));
 | |
|     if (StringEqual (name, "dSizeY"))
 | |
|         return (PyFloat_FromDouble(object->dsize[1]));
 | |
|     if (StringEqual (name, "dSizeZ"))
 | |
|         return (PyFloat_FromDouble(object->dsize[2]));
 | |
|     if (StringEqual (name, "dsize"))
 | |
|         return (Py_BuildValue ("fff", object->dsize[0], object->dsize[1],
 | |
|                                object->dsize[2]));
 | |
|     if (strncmp (name,"Eff", 3) == 0)
 | |
|     {
 | |
|         if ( (object->type == OB_IKA) && (object->data != NULL) )
 | |
|         {
 | |
|             ika = object->data;
 | |
|             switch (name[3])
 | |
|             {
 | |
|                 case 'X':
 | |
|                     return (PyFloat_FromDouble (ika->effg[0]));
 | |
|                 case 'Y':
 | |
|                     return (PyFloat_FromDouble (ika->effg[1]));
 | |
|                 case 'Z':
 | |
|                     return (PyFloat_FromDouble (ika->effg[2]));
 | |
| 						default:
 | |
| 							/* Do we need to display a sensible error message here? */
 | |
| 							return (NULL);
 | |
|             }
 | |
|         }
 | |
|         return (NULL);
 | |
|     }
 | |
|     if (StringEqual (name, "Layer"))
 | |
| 			return (PyInt_FromLong(object->lay));
 | |
|     if (StringEqual (name, "parent"))
 | |
|     {
 | |
|         if (object->parent)
 | |
|             return (Object_CreatePyObject (object->parent));
 | |
|         else 
 | |
| 		{
 | |
| 			Py_INCREF (Py_None);
 | |
| 			return (Py_None);
 | |
| 		}
 | |
|     }
 | |
| 
 | |
|     if (StringEqual (name, "track"))
 | |
| 			return (Object_CreatePyObject (object->track));
 | |
|     if (StringEqual (name, "data"))
 | |
| 			return (Object_getData (obj));
 | |
|     if (StringEqual (name, "ipo"))
 | |
|     {
 | |
|         if (object->ipo == NULL)
 | |
|         {
 | |
|             /* There's no ipo in the object, so we need to return Py_None */
 | |
|             /* However, if there's already some kind of reference in the */
 | |
|             /* Python object, we also need to free that one. */
 | |
|             if (obj->ipo != NULL)
 | |
|             {
 | |
|                 Py_DECREF (obj->ipo);
 | |
|             }
 | |
| 
 | |
|             /* There's no ipo, so this needs to be NULL always. */
 | |
|             obj->ipo = NULL;
 | |
| 
 | |
|             Py_INCREF (Py_None);
 | |
|             return (Py_None);
 | |
|         }
 | |
|         if (obj->ipo == NULL)
 | |
|         {
 | |
|             if (object->ipo == NULL)
 | |
|             {
 | |
|                 /* There's no ipo linked to the object, return Py_None. */
 | |
|                 Py_INCREF (Py_None);
 | |
|                 return (Py_None);
 | |
|             }
 | |
|             obj->ipo = Ipo_CreatePyObject (object->ipo);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if (Ipo_FromPyObject (obj->ipo) != object->ipo)
 | |
|             {
 | |
|                 /* The ipo object has changed, so decref the original */
 | |
|                 /* py_ipo object, and create a new one. */
 | |
|                 Py_DECREF (obj->ipo);
 | |
|                 obj->ipo = Ipo_CreatePyObject (object->ipo);
 | |
|             }
 | |
|         }
 | |
|         Py_INCREF (obj->ipo);
 | |
|         return (obj->ipo);
 | |
|     }
 | |
|     if (StringEqual (name, "mat"))
 | |
|         return (Object_getMatrix (obj));
 | |
|     if (StringEqual (name, "matrix"))
 | |
|         return (Object_getMatrix (obj));
 | |
|     if (StringEqual (name, "colbits"))
 | |
|         return (Py_BuildValue ("h", object->colbits));
 | |
|     if (StringEqual (name, "drawType"))
 | |
|         return (Py_BuildValue ("b", object->dt));
 | |
|     if (StringEqual (name, "drawMode"))
 | |
|         return (Py_BuildValue ("b", object->dtx));
 | |
|     if (StringEqual (name, "name"))
 | |
|         return (Py_BuildValue ("s", object->id.name+2));
 | |
| 
 | |
|     /* not an attribute, search the methods table */
 | |
|     return Py_FindMethod(BPy_Object_methods, (PyObject *)obj, name);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_setAttr                                               */
 | |
| /* Description: This is a callback function for the BlenObject type. It is   */
 | |
| /*              the function that retrieves any value from Python and sets   */
 | |
| /*              it accordingly in Blender.                                   */
 | |
| /*****************************************************************************/
 | |
| static int Object_setAttr (BPy_Object *obj, char *name, PyObject *value)
 | |
| {
 | |
|     struct Object    * object;
 | |
|     struct Ika      * ika;
 | |
| 
 | |
|     object = obj->object;
 | |
|     if (StringEqual (name, "LocX"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->loc[0])));
 | |
|     if (StringEqual (name, "LocY"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->loc[1])));
 | |
|     if (StringEqual (name, "LocZ"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->loc[2])));
 | |
|     if (StringEqual (name, "loc"))
 | |
|     {
 | |
|         if (Object_setLocation (obj, value) != Py_None)
 | |
|             return (-1);
 | |
|         else
 | |
|             return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "dLocX"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->dloc[0])));
 | |
|     if (StringEqual (name, "dLocY"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->dloc[1])));
 | |
|     if (StringEqual (name, "dLocZ"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->dloc[2])));
 | |
|     if (StringEqual (name, "dloc"))
 | |
|     {
 | |
|         if (Object_setDeltaLocation (obj, value) != Py_None)
 | |
|             return (-1);
 | |
|         else
 | |
|             return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "RotX"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->rot[0])));
 | |
|     if (StringEqual (name, "RotY"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->rot[1])));
 | |
|     if (StringEqual (name, "RotZ"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->rot[2])));
 | |
|     if (StringEqual (name, "rot"))
 | |
|     {
 | |
|         if (Object_setEuler (obj, value) != Py_None)
 | |
|             return (-1);
 | |
|         else
 | |
|             return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "dRotX"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->drot[0])));
 | |
|     if (StringEqual (name, "dRotY"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->drot[1])));
 | |
|     if (StringEqual (name, "dRotZ"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->drot[2])));
 | |
|     if (StringEqual (name, "drot"))
 | |
|         return (!PyArg_Parse (value, "fff", &(object->drot[0]),
 | |
|                               &(object->drot[1]), &(object->drot[2])));
 | |
|     if (StringEqual (name, "SizeX"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->size[0])));
 | |
|     if (StringEqual (name, "SizeY"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->size[1])));
 | |
|     if (StringEqual (name, "SizeZ"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->size[2])));
 | |
|     if (StringEqual (name, "size"))
 | |
|         return (!PyArg_Parse (value, "fff", &(object->size[0]),
 | |
|                               &(object->size[1]), &(object->size[2])));
 | |
|     if (StringEqual (name, "dSizeX"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->dsize[0])));
 | |
|     if (StringEqual (name, "dSizeY"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->dsize[1])));
 | |
|     if (StringEqual (name, "dSizeZ"))
 | |
|         return (!PyArg_Parse (value, "f", &(object->dsize[2])));
 | |
|     if (StringEqual (name, "dsize"))
 | |
|         return (!PyArg_Parse (value, "fff", &(object->dsize[0]),
 | |
|                               &(object->dsize[1]), &(object->dsize[2])));
 | |
|     if (strncmp (name,"Eff", 3) == 0)
 | |
|     {
 | |
|         if ( (object->type == OB_IKA) && (object->data != NULL) )
 | |
|         {
 | |
|             ika = object->data;
 | |
|             switch (name[3])
 | |
|             {
 | |
|                 case 'X':
 | |
|                     return (!PyArg_Parse (value, "f", &(ika->effg[0])));
 | |
|                 case 'Y':
 | |
|                     return (!PyArg_Parse (value, "f", &(ika->effg[1])));
 | |
|                 case 'Z':
 | |
|                     return (!PyArg_Parse (value, "f", &(ika->effg[2])));
 | |
|                 default:
 | |
|                     /* Do we need to display a sensible error message here? */
 | |
|                     return (0);
 | |
|             }
 | |
|         }
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "Layer"))
 | |
|         return (!PyArg_Parse (value, "i", &(object->lay)));
 | |
|     if (StringEqual (name, "parent"))
 | |
|     {
 | |
|         /* This is not allowed. */
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "Setting the parent is not allowed.");
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "track"))
 | |
|     {
 | |
|         /* This is not allowed. */
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "Setting the track is not allowed.");
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "data"))
 | |
|     {
 | |
|         /* This is not allowed. */
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "Setting the data is not allowed.");
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "ipo"))
 | |
|     {
 | |
|         /* This is not allowed. */
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "Setting the ipo is not allowed.");
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "mat"))
 | |
|     {
 | |
|         /* This is not allowed. */
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "Setting the matrix is not allowed.");
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "matrix"))
 | |
|     {
 | |
|         /* This is not allowed. */
 | |
|         PythonReturnErrorObject (PyExc_AttributeError,
 | |
|                     "Setting the matrix is not allowed.");
 | |
|         return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "colbits"))
 | |
|         return (!PyArg_Parse (value, "h", &(object->colbits)));
 | |
|     if (StringEqual (name, "drawType"))
 | |
|     {
 | |
|         if (Object_setDrawType (obj, value) != Py_None)
 | |
|             return (-1);
 | |
|         else
 | |
|             return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "drawMode"))
 | |
|     {
 | |
|         if (Object_setDrawMode (obj, value) != Py_None)
 | |
|             return (-1);
 | |
|         else
 | |
|             return (0);
 | |
|     }
 | |
|     if (StringEqual (name, "name"))
 | |
|     {
 | |
|         if (Object_setName (obj, value) != Py_None)
 | |
|             return (-1);
 | |
|         else
 | |
|             return (0);
 | |
|     }
 | |
| 
 | |
|     printf ("Unknown variable.\n");
 | |
|     return (0);
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_compare                                               */
 | |
| /* Description: This is a callback function for the BPy_Object type. It      */
 | |
| /*              compares two Object_Type objects. Only the "==" and "!="     */
 | |
| /*              comparisons are meaninful. Returns 0 for equality and -1 if  */
 | |
| /*              they don't point to the same Blender Object struct.          */
 | |
| /*              In Python it becomes 1 if they are equal, 0 otherwise.       */
 | |
| /*****************************************************************************/
 | |
| static int Object_compare (BPy_Object *a, BPy_Object *b)
 | |
| {
 | |
|   Object *pa = a->object, *pb = b->object;
 | |
|   return (pa == pb) ? 0:-1;
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /* Function:    Object_repr                                                  */
 | |
| /* Description: This is a callback function for the BPy_Object type. It      */
 | |
| /*              builds a meaninful string to represent object objects.       */
 | |
| /*****************************************************************************/
 | |
| static PyObject *Object_repr (BPy_Object *self)
 | |
| {
 | |
|   return PyString_FromFormat("[Object \"%s\"]", self->object->id.name+2);
 | |
| }
 | |
| 
 |