Ongoing updates to the Blender.Node Python API: - Changed Blender.Node.node to Blender.Node.Scripted to be more specific and conform to bpython API. - Added a Blender.Node.Socket type to be used to define node sockets in a PyNode script. Also, socket type is inferred from the value(s) passed in, instead of also being defined by the script author. - Added attr access to input and output sockets in the __call__ method. Ex: an input socket called 'color' can be accessed as self.input.color now. These changes break existing pynode scripts, which shouldn't be a problem yet, of course, since we're still finishing this feature for 2.46. The wiki page and sample .blends have already been updated: http://wiki.blender.org/index.php/BlenderDev/PyNodes http://wiki.blender.org/index.php/BlenderDev/PyNodes/API
		
			
				
	
	
		
			1622 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1622 lines
		
	
	
		
			51 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) 2006, Blender Foundation
 | 
						|
 * All rights reserved.
 | 
						|
 *
 | 
						|
 * Original code is this file
 | 
						|
 *
 | 
						|
 * Contributor(s): Nathan Letwory
 | 
						|
 *
 | 
						|
 * ***** END GPL/BL DUAL LICENSE BLOCK *****
 | 
						|
*/
 | 
						|
 | 
						|
#include "Node.h"
 | 
						|
 | 
						|
#include "BKE_global.h"
 | 
						|
#include "BKE_main.h"
 | 
						|
#include "BKE_node.h"
 | 
						|
#include "BKE_utildefines.h"
 | 
						|
 | 
						|
#include "DNA_material_types.h"
 | 
						|
 | 
						|
#include "MEM_guardedalloc.h"
 | 
						|
 | 
						|
#include "BLI_blenlib.h"
 | 
						|
#include "gen_utils.h"
 | 
						|
#include "vector.h"
 | 
						|
 | 
						|
static PyObject *Node_repr( BPy_Node * self );
 | 
						|
static int Node_compare(BPy_Node *a, BPy_Node *b);
 | 
						|
static PyObject *ShadeInput_repr( BPy_ShadeInput * self );
 | 
						|
static int ShadeInput_compare(BPy_ShadeInput *a, BPy_ShadeInput *b);
 | 
						|
static BPy_ShadeInput *ShadeInput_CreatePyObject(ShadeInput *shi);
 | 
						|
 | 
						|
/* node socket type */
 | 
						|
 | 
						|
static PyObject *NodeSocket_getName(BPy_NodeSocket *self, void *unused)
 | 
						|
{
 | 
						|
	return PyString_FromString(self->name);
 | 
						|
}
 | 
						|
 | 
						|
static int NodeSocket_setName(BPy_NodeSocket *self, PyObject *value, void *unused)
 | 
						|
{
 | 
						|
	char *name = NULL;
 | 
						|
 | 
						|
	if (!PyString_Check(value))
 | 
						|
		return EXPP_ReturnIntError( PyExc_TypeError,
 | 
						|
			"expected a string" );
 | 
						|
 | 
						|
	name = PyString_AsString(value);
 | 
						|
 | 
						|
	if (!name)
 | 
						|
		return EXPP_ReturnIntError(PyExc_RuntimeError,
 | 
						|
			"couldn't convert value to string!");
 | 
						|
 | 
						|
	BLI_strncpy(self->name, name, NODE_MAXSTR);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *NodeSocket_getVal(BPy_NodeSocket *self, void *unused)
 | 
						|
{
 | 
						|
	PyObject *pyret = NULL;
 | 
						|
 | 
						|
	if (self->type == SOCK_VALUE) {
 | 
						|
		pyret = PyFloat_FromDouble(self->val[0]);
 | 
						|
	}
 | 
						|
	else { /* VECTOR or RGBA */
 | 
						|
		pyret = newVectorObject(self->val, self->type, Py_NEW);
 | 
						|
 | 
						|
		if (!pyret)
 | 
						|
			return EXPP_ReturnPyObjError(PyExc_RuntimeError,
 | 
						|
				"couldn't create vector object!");
 | 
						|
	}
 | 
						|
 | 
						|
	return pyret;
 | 
						|
}
 | 
						|
 | 
						|
static int NodeSocket_setVal(BPy_NodeSocket *self, PyObject *value, void *unused)
 | 
						|
{
 | 
						|
	int error = 0;
 | 
						|
 | 
						|
	if (PySequence_Check(value)) {
 | 
						|
		PyObject *item, *fpyval;
 | 
						|
		int i, len;
 | 
						|
 | 
						|
		len = PySequence_Size(value);
 | 
						|
 | 
						|
		if (len == 3 || len == 4) {
 | 
						|
			for (i = 0; i < len; i++) {
 | 
						|
				item = PySequence_GetItem(value, i);
 | 
						|
				fpyval = PyNumber_Float(item);
 | 
						|
				if (!fpyval) {
 | 
						|
					Py_DECREF(item);
 | 
						|
					error = 1;
 | 
						|
					break;
 | 
						|
				}
 | 
						|
				self->val[i] = (float)PyFloat_AsDouble(fpyval);
 | 
						|
				Py_DECREF(item);
 | 
						|
				Py_DECREF(fpyval);
 | 
						|
			}
 | 
						|
			if (len == 3)
 | 
						|
				self->type = SOCK_VECTOR;
 | 
						|
			else /* len == 4 */
 | 
						|
				self->type = SOCK_RGBA;
 | 
						|
		}
 | 
						|
		else error = 1;
 | 
						|
	}
 | 
						|
	else if (VectorObject_Check(value)) {
 | 
						|
		VectorObject *vecOb = (VectorObject *)value;
 | 
						|
		short vlen = vecOb->size;
 | 
						|
 | 
						|
		if (vlen == 3 || vlen == 4) {
 | 
						|
			VECCOPY(self->val, vecOb->vec); /* copies 3 values */
 | 
						|
			if (vlen == 3)
 | 
						|
				self->type = SOCK_VECTOR;
 | 
						|
			else {
 | 
						|
				self->val[3] = vecOb->vec[3];
 | 
						|
				self->type = SOCK_RGBA;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else error = 1;
 | 
						|
	}
 | 
						|
	else if (PyNumber_Check(value)) {
 | 
						|
		self->val[0] = (float)PyFloat_AsDouble(value);
 | 
						|
		self->type = SOCK_VALUE;
 | 
						|
	}
 | 
						|
	else error = 1;
 | 
						|
 | 
						|
	if (error)
 | 
						|
		return EXPP_ReturnIntError( PyExc_TypeError,
 | 
						|
			"expected a float or a sequence (or vector) of 3 or 4 floats" );
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *NodeSocket_getMin(BPy_NodeSocket *self, void *unused)
 | 
						|
{
 | 
						|
	return PyFloat_FromDouble(self->min);
 | 
						|
}
 | 
						|
 | 
						|
static int NodeSocket_setMin(BPy_NodeSocket *self, PyObject *value, void *unused)
 | 
						|
{
 | 
						|
	PyObject *pyval = PyNumber_Float(value);
 | 
						|
 | 
						|
	if (!pyval)
 | 
						|
		return EXPP_ReturnIntError( PyExc_TypeError,
 | 
						|
			"expected a float number" );
 | 
						|
 | 
						|
	self->min = (float)PyFloat_AsDouble(pyval);
 | 
						|
	Py_DECREF(pyval);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *NodeSocket_getMax(BPy_NodeSocket *self, void *unused)
 | 
						|
{
 | 
						|
	return PyFloat_FromDouble(self->max);
 | 
						|
}
 | 
						|
 | 
						|
static int NodeSocket_setMax(BPy_NodeSocket *self, PyObject *value, void *unused)
 | 
						|
{
 | 
						|
	PyObject *pyval = PyNumber_Float(value);
 | 
						|
 | 
						|
	if (!pyval)
 | 
						|
		return EXPP_ReturnIntError( PyExc_TypeError,
 | 
						|
			"expected a float number" );
 | 
						|
 | 
						|
	self->max = (float)PyFloat_AsDouble(pyval);
 | 
						|
	Py_DECREF(pyval);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyGetSetDef NodeSocket_getseters[] = {
 | 
						|
	{"name", (getter)NodeSocket_getName, (setter)NodeSocket_setName,
 | 
						|
		"This socket's name", NULL},
 | 
						|
	{"val", (getter)NodeSocket_getVal, (setter)NodeSocket_setVal,
 | 
						|
		"This socket's data value(s)", NULL},
 | 
						|
	{"min", (getter)NodeSocket_getMin, (setter)NodeSocket_setMin,
 | 
						|
		"This socket's min possible value (lower range limit)", NULL},
 | 
						|
	{"max", (getter)NodeSocket_getMax, (setter)NodeSocket_setMax,
 | 
						|
		"This socket's max possible value (upper range limit)", NULL},
 | 
						|
	{NULL,NULL,NULL,NULL,NULL}  /* Sentinel */
 | 
						|
};
 | 
						|
 | 
						|
static void NodeSocket_dealloc(BPy_NodeSocket *self)
 | 
						|
{
 | 
						|
	self->ob_type->tp_free((PyObject *)self);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *NodeSocket_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | 
						|
{
 | 
						|
	PyObject *pysocket = type->tp_alloc(type, 0);
 | 
						|
 | 
						|
	if (!pysocket)
 | 
						|
		return EXPP_ReturnPyObjError(PyExc_RuntimeError, "couldn't create socket type!");
 | 
						|
 | 
						|
	return pysocket;
 | 
						|
}
 | 
						|
 | 
						|
static int NodeSocket_init(BPy_NodeSocket *self, PyObject *args, PyObject *kwargs)
 | 
						|
{
 | 
						|
	char *name = NULL;
 | 
						|
	float min = 0.0f, max = 1.0f;
 | 
						|
	PyObject *val = NULL;
 | 
						|
	static char *kwlist[] = {"name", "val", "min", "max", NULL};
 | 
						|
 | 
						|
	if (!PyArg_ParseTupleAndKeywords(args, kwargs, "s|Off", kwlist, &name, &val, &min, &max)){
 | 
						|
		return EXPP_ReturnIntError(PyExc_AttributeError, "expected a string and optionally:\n1) a float or a sequence (or vector) of 3 or 4 floats and\n2) two floats");
 | 
						|
	}
 | 
						|
 | 
						|
	BLI_strncpy(self->name, name, NODE_MAXSTR);
 | 
						|
 | 
						|
	self->min = min;
 | 
						|
	self->max = max;
 | 
						|
 | 
						|
	if (val)
 | 
						|
		return NodeSocket_setVal(self, val, NULL);
 | 
						|
	/* else */
 | 
						|
	self->type = SOCK_VALUE;
 | 
						|
	self->val[0] = 0.0f;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *NodeSocket_copy(BPy_NodeSocket *self)
 | 
						|
{
 | 
						|
	BPy_NodeSocket *copied;
 | 
						|
 | 
						|
	copied = (BPy_NodeSocket*)NodeSocket_new(&NodeSocket_Type, NULL, NULL);
 | 
						|
 | 
						|
	if (!copied) return NULL; /* error already set in NodeSocket_new */
 | 
						|
 | 
						|
	BLI_strncpy(copied->name, self->name, NODE_MAXSTR);
 | 
						|
	copied->min = self->min;
 | 
						|
	copied->max = self->max;
 | 
						|
	copied->type = self->type;
 | 
						|
	memcpy(copied->val, self->val, 4*sizeof(float));
 | 
						|
 | 
						|
	return (PyObject *)copied;
 | 
						|
}
 | 
						|
 | 
						|
static PyMethodDef BPy_NodeSocket_methods[] = {
 | 
						|
	{"__copy__", ( PyCFunction ) NodeSocket_copy, METH_NOARGS,
 | 
						|
	 "() - Makes a copy of this node socket."},
 | 
						|
	{"copy", ( PyCFunction ) NodeSocket_copy, METH_NOARGS,
 | 
						|
	 "() - Makes a copy of this node socket."},
 | 
						|
	{NULL, NULL, 0, NULL}
 | 
						|
};
 | 
						|
 | 
						|
PyTypeObject NodeSocket_Type = {
 | 
						|
	PyObject_HEAD_INIT( NULL )  /* required py macro */
 | 
						|
	0,                          /* ob_size */
 | 
						|
	/*  For printing, in format "<module>.<name>" */
 | 
						|
	"Blender.Node.Socket",           /* char *tp_name; */
 | 
						|
	sizeof( BPy_NodeSocket ),       /* int tp_basicsize; */
 | 
						|
	0,                          /* tp_itemsize;  For allocation */
 | 
						|
 | 
						|
	/* Methods to implement standard operations */
 | 
						|
 | 
						|
	(destructor)NodeSocket_dealloc,/* destructor tp_dealloc; */
 | 
						|
	NULL,                       /* printfunc tp_print; */
 | 
						|
	NULL,                       /* getattrfunc tp_getattr; */
 | 
						|
	NULL,                       /* setattrfunc tp_setattr; */
 | 
						|
	NULL,                       /* cmpfunc tp_compare; */
 | 
						|
	NULL,                       /* reprfunc tp_repr; */
 | 
						|
 | 
						|
	/* Method suites for standard classes */
 | 
						|
 | 
						|
	NULL,      					/* PyNumberMethods *tp_as_number; */
 | 
						|
	NULL,					    /* PySequenceMethods *tp_as_sequence; */
 | 
						|
	NULL,      /* PyMappingMethods *tp_as_mapping; */
 | 
						|
 | 
						|
	/* More standard operations (here for binary compatibility) */
 | 
						|
 | 
						|
	NULL,                       /* hashfunc tp_hash; */
 | 
						|
	NULL,                       /* ternaryfunc tp_call; */
 | 
						|
	NULL,                       /* reprfunc tp_str; */
 | 
						|
	NULL,                       /* getattrofunc tp_getattro; */
 | 
						|
	NULL,                       /* setattrofunc tp_setattro; */
 | 
						|
 | 
						|
	/* Functions to access object as input/input buffer */
 | 
						|
	NULL,                       /* PyBufferProcs *tp_as_buffer; */
 | 
						|
 | 
						|
  /*** Flags to define presence of optional/expanded features ***/
 | 
						|
	Py_TPFLAGS_DEFAULT|Py_TPFLAGS_BASETYPE, /* long tp_flags; */
 | 
						|
 | 
						|
	NULL,                       /*  char *tp_doc;  Documentation string */
 | 
						|
  /*** Assigned meaning in release 2.0 ***/
 | 
						|
	/* call function for all accessible objects */
 | 
						|
	NULL,                       /* traverseproc tp_traverse; */
 | 
						|
 | 
						|
	/* delete references to contained objects */
 | 
						|
	NULL,                       /* inquiry tp_clear; */
 | 
						|
 | 
						|
  /***  Assigned meaning in release 2.1 ***/
 | 
						|
  /*** rich comparisons ***/
 | 
						|
	NULL,                       /* richcmpfunc tp_richcompare; */
 | 
						|
 | 
						|
  /***  weak reference enabler ***/
 | 
						|
	0,                          /* long tp_weaklistoffset; */
 | 
						|
 | 
						|
  /*** Added in release 2.2 ***/
 | 
						|
	/*   Iterators */
 | 
						|
	0, //( getiterfunc) MVertSeq_getIter, /* getiterfunc tp_iter; */
 | 
						|
	0, //( iternextfunc ) MVertSeq_nextIter, /* iternextfunc tp_iternext; */
 | 
						|
 | 
						|
  /*** Attribute descriptor and subclassing stuff ***/
 | 
						|
	BPy_NodeSocket_methods,     /* struct PyMethodDef *tp_methods; */
 | 
						|
	NULL,                       /* struct PyMemberDef *tp_members; */
 | 
						|
	NodeSocket_getseters,       /* struct PyGetSetDef *tp_getset; */
 | 
						|
	NULL,                       /* struct _typeobject *tp_base; */
 | 
						|
	NULL,                       /* PyObject *tp_dict; */
 | 
						|
	NULL,                       /* descrgetfunc tp_descr_get; */
 | 
						|
	NULL,                       /* descrsetfunc tp_descr_set; */
 | 
						|
	0,                          /* long tp_dictoffset; */
 | 
						|
	(initproc)NodeSocket_init,  /* initproc tp_init; */
 | 
						|
	NULL,                       /* allocfunc tp_alloc; */
 | 
						|
	NodeSocket_new,				/* newfunc tp_new; */
 | 
						|
	/*  Low-level free-memory routine */
 | 
						|
	NULL,                       /* freefunc tp_free;  */
 | 
						|
	/* For PyObject_IS_GC */
 | 
						|
	NULL,                       /* inquiry tp_is_gc;  */
 | 
						|
	NULL,                       /* PyObject *tp_bases; */
 | 
						|
	/* method resolution order */
 | 
						|
	NULL,                       /* PyObject *tp_mro;  */
 | 
						|
	NULL,                       /* PyObject *tp_cache; */
 | 
						|
	NULL,                       /* PyObject *tp_subclasses; */
 | 
						|
	NULL,                       /* PyObject *tp_weaklist; */
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Take the descriptions from tuple and create sockets for those in socks
 | 
						|
 * socks is a socketstack from a bNodeTypeInfo
 | 
						|
 */
 | 
						|
static int pysockets_to_blendersockets(PyObject *tuple, bNodeSocketType **socks, int stage, int limit) {
 | 
						|
	int len = 0, a = 0, pos = 0, retval = 0;
 | 
						|
	short stype;
 | 
						|
	BPy_NodeSocket *pysock;
 | 
						|
	bNodeSocketType *nsocks = NULL;
 | 
						|
 | 
						|
	if (BTST2(stage, NODE_DYNAMIC_READY, NODE_DYNAMIC_ADDEXIST))
 | 
						|
		return 0; /* already has sockets */
 | 
						|
 | 
						|
	len = PyTuple_Size(tuple);
 | 
						|
 | 
						|
	nsocks = MEM_callocN(sizeof(bNodeSocketType)*(len+1), "bNodeSocketType in Node.c");
 | 
						|
 | 
						|
	for (pos = 0, a = 0; pos< len; pos++, a++) {
 | 
						|
		pysock = (BPy_NodeSocket *)PyTuple_GetItem(tuple, pos);/*borrowed*/
 | 
						|
 | 
						|
		if (!BPy_NodeSocket_Check(pysock)) {
 | 
						|
			PyErr_SetString(PyExc_AttributeError, "expected a sequence of node sockets");
 | 
						|
			retval = -1;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		stype = pysock->type;
 | 
						|
 | 
						|
		nsocks[a].type = stype;
 | 
						|
		nsocks[a].limit = limit;
 | 
						|
 | 
						|
		nsocks[a].name = BLI_strdupn(pysock->name, NODE_MAXSTR);
 | 
						|
 | 
						|
		nsocks[a].min = pysock->min;
 | 
						|
		nsocks[a].max = pysock->max;
 | 
						|
 | 
						|
		if (stype > SOCK_VALUE) {
 | 
						|
			float *vec = pysock->val;
 | 
						|
 | 
						|
			nsocks[a].val1 = vec[0];
 | 
						|
			nsocks[a].val2 = vec[1];
 | 
						|
			nsocks[a].val3 = vec[2];
 | 
						|
 | 
						|
			if (stype == SOCK_RGBA)
 | 
						|
				nsocks[a].val4 = vec[3];
 | 
						|
		}
 | 
						|
		else /* SOCK_VALUE */
 | 
						|
			nsocks[a].val1 = pysock->val[0];
 | 
						|
	}
 | 
						|
 | 
						|
	nsocks[a].type = -1;
 | 
						|
 | 
						|
	*socks = nsocks;
 | 
						|
 | 
						|
	return retval;
 | 
						|
}
 | 
						|
 | 
						|
static void NodeSocketLists_dealloc(BPy_NodeSocketLists *self)
 | 
						|
{
 | 
						|
	Py_DECREF(self->input);
 | 
						|
	Py_DECREF(self->output);
 | 
						|
	self->ob_type->tp_free((PyObject *)self);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *Map_socketdef_getter(BPy_NodeSocketLists *self, void *closure)
 | 
						|
{
 | 
						|
	PyObject *sockets = NULL;
 | 
						|
 | 
						|
	switch ((int)closure) {
 | 
						|
		case 'I': /* inputs */
 | 
						|
			Py_INCREF(self->input);
 | 
						|
			sockets = self->input;
 | 
						|
			break;
 | 
						|
		case 'O': /* outputs */
 | 
						|
			Py_INCREF(self->output);
 | 
						|
			sockets = self->output;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			fprintf(stderr, "DEBUG pynodes: wrong option in Map_socketdef_getter\n");
 | 
						|
			Py_INCREF(Py_None);
 | 
						|
			sockets = Py_None;
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	return sockets;
 | 
						|
}
 | 
						|
 | 
						|
static int Map_socketdef(BPy_NodeSocketLists *self, PyObject *args, void *closure)
 | 
						|
{
 | 
						|
	bNode *node = NULL;
 | 
						|
	PyObject *tuple = NULL;
 | 
						|
 | 
						|
	node = self->node;
 | 
						|
 | 
						|
	if(!node) {
 | 
						|
		fprintf(stderr,"DEBUG pynodes: No bNode in BPy_Node (Map_socketdef)\n");
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	if(BTST2(node->custom1, NODE_DYNAMIC_READY, NODE_DYNAMIC_ADDEXIST))
 | 
						|
		return 0;
 | 
						|
 | 
						|
	switch((int)closure) {
 | 
						|
		case 'I':
 | 
						|
			if (args) {
 | 
						|
				if(PySequence_Check(args)) {
 | 
						|
					tuple = PySequence_Tuple(args);
 | 
						|
					pysockets_to_blendersockets(tuple,
 | 
						|
							&(node->typeinfo->inputs), node->custom1, 1);
 | 
						|
					Py_DECREF(self->input);
 | 
						|
					self->input = tuple;
 | 
						|
				} else {
 | 
						|
					return(EXPP_ReturnIntError( PyExc_AttributeError, "INPUT must be a List of Lists or Tuples"));
 | 
						|
				}
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case 'O':
 | 
						|
			if (args) {
 | 
						|
				if(PyList_Check(args)) {
 | 
						|
					tuple = PySequence_Tuple(args);
 | 
						|
					pysockets_to_blendersockets(tuple,
 | 
						|
							&(node->typeinfo->outputs), node->custom1, 0);
 | 
						|
					Py_DECREF(self->output);
 | 
						|
					self->output = tuple;
 | 
						|
				} else {
 | 
						|
					return(EXPP_ReturnIntError( PyExc_AttributeError, "OUTPUT must be a List of Lists or Tuples"));
 | 
						|
				}
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			fprintf(stderr,"DEBUG pynodes: got no list in Map_socketdef\n");
 | 
						|
			break;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyGetSetDef NodeSocketLists_getseters[] = {
 | 
						|
	{"input", (getter)Map_socketdef_getter, (setter)Map_socketdef,
 | 
						|
		"Set this node's input sockets (list of lists or tuples)",
 | 
						|
		(void *)'I'},
 | 
						|
	{"i" /*alias*/, (getter)Map_socketdef_getter, (setter)Map_socketdef,
 | 
						|
		"Set this node's input sockets (list of lists or tuples)",
 | 
						|
		(void *)'I'},
 | 
						|
	{"output", (getter)Map_socketdef_getter, (setter)Map_socketdef,
 | 
						|
		"Set this node's output sockets (list of lists or tuples)",
 | 
						|
		(void *)'O'},
 | 
						|
	{"o" /*alias*/, (getter)Map_socketdef_getter, (setter)Map_socketdef,
 | 
						|
		"Set this node's output sockets (list of lists or tuples)",
 | 
						|
		(void *)'O'},
 | 
						|
	{NULL,NULL,NULL,NULL,NULL}  /* Sentinel */
 | 
						|
};
 | 
						|
 | 
						|
PyTypeObject NodeSocketLists_Type = {
 | 
						|
	PyObject_HEAD_INIT( NULL )  /* required py macro */
 | 
						|
	0,                          /* ob_size */
 | 
						|
	/*  For printing, in format "<module>.<name>" */
 | 
						|
	"Blender.Node.SocketLists",           /* char *tp_name; */
 | 
						|
	sizeof( BPy_NodeSocketLists ),       /* int tp_basicsize; */
 | 
						|
	0,                          /* tp_itemsize;  For allocation */
 | 
						|
 | 
						|
	/* Methods to implement standard operations */
 | 
						|
 | 
						|
	(destructor)NodeSocketLists_dealloc,/* destructor tp_dealloc; */
 | 
						|
	NULL,                       /* printfunc tp_print; */
 | 
						|
	NULL,                       /* getattrfunc tp_getattr; */
 | 
						|
	NULL,                       /* setattrfunc tp_setattr; */
 | 
						|
	NULL,                       /* cmpfunc tp_compare; */
 | 
						|
	NULL,                       /* reprfunc tp_repr; */
 | 
						|
 | 
						|
	/* Method suites for standard classes */
 | 
						|
 | 
						|
	NULL,      					/* PyNumberMethods *tp_as_number; */
 | 
						|
	NULL,					    /* PySequenceMethods *tp_as_sequence; */
 | 
						|
	NULL,      /* PyMappingMethods *tp_as_mapping; */
 | 
						|
 | 
						|
	/* More standard operations (here for binary compatibility) */
 | 
						|
 | 
						|
	NULL,                       /* hashfunc tp_hash; */
 | 
						|
	NULL,                       /* ternaryfunc tp_call; */
 | 
						|
	NULL,                       /* reprfunc tp_str; */
 | 
						|
	NULL,                       /* getattrofunc tp_getattro; */
 | 
						|
	NULL,                       /* setattrofunc tp_setattro; */
 | 
						|
 | 
						|
	/* Functions to access object as input/input buffer */
 | 
						|
	NULL,                       /* PyBufferProcs *tp_as_buffer; */
 | 
						|
 | 
						|
  /*** Flags to define presence of optional/expanded features ***/
 | 
						|
	Py_TPFLAGS_DEFAULT,         /* long tp_flags; */
 | 
						|
 | 
						|
	NULL,                       /*  char *tp_doc;  Documentation string */
 | 
						|
  /*** Assigned meaning in release 2.0 ***/
 | 
						|
	/* call function for all accessible objects */
 | 
						|
	NULL,                       /* traverseproc tp_traverse; */
 | 
						|
 | 
						|
	/* delete references to contained objects */
 | 
						|
	NULL,                       /* inquiry tp_clear; */
 | 
						|
 | 
						|
  /***  Assigned meaning in release 2.1 ***/
 | 
						|
  /*** rich comparisons ***/
 | 
						|
	NULL,                       /* richcmpfunc tp_richcompare; */
 | 
						|
 | 
						|
  /***  weak reference enabler ***/
 | 
						|
	0,                          /* long tp_weaklistoffset; */
 | 
						|
 | 
						|
  /*** Added in release 2.2 ***/
 | 
						|
	/*   Iterators */
 | 
						|
	0, //( getiterfunc) MVertSeq_getIter, /* getiterfunc tp_iter; */
 | 
						|
	0, //( iternextfunc ) MVertSeq_nextIter, /* iternextfunc tp_iternext; */
 | 
						|
 | 
						|
  /*** Attribute descriptor and subclassing stuff ***/
 | 
						|
	0, //BPy_MVertSeq_methods,       /* struct PyMethodDef *tp_methods; */
 | 
						|
	NULL,                       /* struct PyMemberDef *tp_members; */
 | 
						|
	NodeSocketLists_getseters,      /* struct PyGetSetDef *tp_getset; */
 | 
						|
	NULL,                       /* struct _typeobject *tp_base; */
 | 
						|
	NULL,                       /* PyObject *tp_dict; */
 | 
						|
	NULL,                       /* descrgetfunc tp_descr_get; */
 | 
						|
	NULL,                       /* descrsetfunc tp_descr_set; */
 | 
						|
	0,                          /* long tp_dictoffset; */
 | 
						|
	NULL,                       /* initproc tp_init; */
 | 
						|
	NULL,                       /* allocfunc tp_alloc; */
 | 
						|
	NULL,                       /* newfunc tp_new; */
 | 
						|
	/*  Low-level free-memory routine */
 | 
						|
	NULL,                       /* freefunc tp_free;  */
 | 
						|
	/* For PyObject_IS_GC */
 | 
						|
	NULL,                       /* inquiry tp_is_gc;  */
 | 
						|
	NULL,                       /* PyObject *tp_bases; */
 | 
						|
	/* method resolution order */
 | 
						|
	NULL,                       /* PyObject *tp_mro;  */
 | 
						|
	NULL,                       /* PyObject *tp_cache; */
 | 
						|
	NULL,                       /* PyObject *tp_subclasses; */
 | 
						|
	NULL,                       /* PyObject *tp_weaklist; */
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
BPy_NodeSocketLists *Node_CreateSocketLists(bNode *node) {
 | 
						|
	BPy_NodeSocketLists *socklists = PyObject_NEW(BPy_NodeSocketLists, &NodeSocketLists_Type);
 | 
						|
	socklists->node = node;
 | 
						|
	socklists->input = PyList_New(0);
 | 
						|
	socklists->output = PyList_New(0);
 | 
						|
	return socklists;
 | 
						|
}
 | 
						|
 | 
						|
/***************************************/
 | 
						|
 | 
						|
static int Sockinmap_len ( BPy_SockMap * self) {
 | 
						|
	bNode *node = self->node;
 | 
						|
	bNodeType *tinfo;
 | 
						|
	int a = 0;
 | 
						|
 | 
						|
	if (!node) return 0;
 | 
						|
 | 
						|
	tinfo = node->typeinfo;
 | 
						|
 | 
						|
	if (BNTST(node->custom1, NODE_DYNAMIC_READY)) return 0;
 | 
						|
 | 
						|
	if (tinfo && tinfo->inputs) {
 | 
						|
		while(self->node->typeinfo->inputs[a].type!=-1)
 | 
						|
			a++;
 | 
						|
	}
 | 
						|
	return a;
 | 
						|
}
 | 
						|
 | 
						|
static int sockinmap_has_key( BPy_SockMap *self, char *strkey) {
 | 
						|
	bNode *node = self->node;
 | 
						|
	bNodeType *tinfo;
 | 
						|
	int a = 0;
 | 
						|
 | 
						|
	if (!node || !strkey) return -1;
 | 
						|
 | 
						|
	tinfo = node->typeinfo;
 | 
						|
 | 
						|
	if(tinfo && tinfo->inputs){
 | 
						|
		while(self->node->typeinfo->inputs[a].type!=-1) {
 | 
						|
			if(BLI_strcaseeq(self->node->typeinfo->inputs[a].name, strkey)) {
 | 
						|
				return a;
 | 
						|
			}
 | 
						|
			a++;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
PyObject *Sockinmap_subscript(BPy_SockMap *self, PyObject *pyidx) {
 | 
						|
	int idx;
 | 
						|
 | 
						|
	if (!self->node)
 | 
						|
		return EXPP_ReturnPyObjError(PyExc_RuntimeError, "no access to Blender node data!");
 | 
						|
 | 
						|
	if (PyString_Check(pyidx)) {
 | 
						|
		idx = sockinmap_has_key(self, PyString_AsString(pyidx));
 | 
						|
	}
 | 
						|
	else if(PyInt_Check(pyidx)) {
 | 
						|
		int len = Sockinmap_len(self);
 | 
						|
		idx = (int)PyInt_AsLong(pyidx);
 | 
						|
		if (idx < 0 || idx >= len)
 | 
						|
			return EXPP_ReturnPyObjError(PyExc_IndexError, "index out of range");
 | 
						|
	}
 | 
						|
	else if (PySlice_Check(pyidx)) {
 | 
						|
		return EXPP_ReturnPyObjError(PyExc_ValueError, "slices not implemented");
 | 
						|
	} else {
 | 
						|
		return EXPP_ReturnPyObjError(PyExc_IndexError, "index must be an int or a string");
 | 
						|
	}
 | 
						|
 | 
						|
	if(idx<0) { /* we're not as nice as Python */
 | 
						|
		return EXPP_ReturnPyObjError(PyExc_IndexError, "invalid socket index");
 | 
						|
	}
 | 
						|
	
 | 
						|
	switch(self->node->typeinfo->inputs[idx].type) {
 | 
						|
		case SOCK_VALUE:
 | 
						|
			return Py_BuildValue("f", self->stack[idx]->vec[0]);
 | 
						|
			break;
 | 
						|
		case SOCK_VECTOR:
 | 
						|
			return Py_BuildValue("(fff)", self->stack[idx]->vec[0], self->stack[idx]->vec[1], self->stack[idx]->vec[2]);
 | 
						|
			break;
 | 
						|
		case SOCK_RGBA:
 | 
						|
			/* otherwise RGBA tuple */
 | 
						|
			return Py_BuildValue("(ffff)", self->stack[idx]->vec[0], self->stack[idx]->vec[1], self->stack[idx]->vec[2], self->stack[idx]->vec[3]);
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	Py_RETURN_NONE;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *Sockinmap_getAttr(BPy_SockMap *self, char *attr)
 | 
						|
{
 | 
						|
	PyObject *pyob = NULL;
 | 
						|
	int idx;
 | 
						|
 | 
						|
	idx = sockinmap_has_key(self, attr);
 | 
						|
 | 
						|
	if (idx < 0)
 | 
						|
		return EXPP_ReturnPyObjError(PyExc_AttributeError, "unknown input socket name");
 | 
						|
 | 
						|
	switch(self->node->typeinfo->inputs[idx].type) {
 | 
						|
		case SOCK_VALUE:
 | 
						|
			pyob = Py_BuildValue("f", self->stack[idx]->vec[0]);
 | 
						|
			break;
 | 
						|
		case SOCK_VECTOR:
 | 
						|
			pyob = Py_BuildValue("(fff)", self->stack[idx]->vec[0], self->stack[idx]->vec[1], self->stack[idx]->vec[2]);
 | 
						|
			break;
 | 
						|
		case SOCK_RGBA:
 | 
						|
			pyob = Py_BuildValue("(ffff)", self->stack[idx]->vec[0], self->stack[idx]->vec[1], self->stack[idx]->vec[2], self->stack[idx]->vec[3]);
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	return pyob;
 | 
						|
}
 | 
						|
 | 
						|
/* read only */
 | 
						|
static PyMappingMethods Sockinmap_as_mapping = {
 | 
						|
	( inquiry ) Sockinmap_len,  /* mp_length */
 | 
						|
	( binaryfunc ) Sockinmap_subscript, /* mp_subscript */
 | 
						|
	( objobjargproc ) 0 /* mp_ass_subscript */
 | 
						|
};
 | 
						|
 | 
						|
PyTypeObject SockInMap_Type = {
 | 
						|
	PyObject_HEAD_INIT( NULL )  /* required py macro */
 | 
						|
	0,                          /* ob_size */
 | 
						|
	/*  For printing, in format "<module>.<name>" */
 | 
						|
	"Blender.Node.InputSockets",           /* char *tp_name; */
 | 
						|
	sizeof( BPy_SockMap ),       /* int tp_basicsize; */
 | 
						|
	0,                          /* tp_itemsize;  For allocation */
 | 
						|
 | 
						|
	/* Methods to implement standard operations */
 | 
						|
 | 
						|
	NULL,/* destructor tp_dealloc; */
 | 
						|
	NULL,                       /* printfunc tp_print; */
 | 
						|
	(getattrfunc) Sockinmap_getAttr,/* getattrfunc tp_getattr; */
 | 
						|
	NULL,                       /* setattrfunc tp_setattr; */
 | 
						|
	NULL,                       /* cmpfunc tp_compare; */
 | 
						|
	NULL,                       /* reprfunc tp_repr; */
 | 
						|
 | 
						|
	/* Method suites for standard classes */
 | 
						|
 | 
						|
	NULL,      					/* PyNumberMethods *tp_as_number; */
 | 
						|
	NULL,					    /* PySequenceMethods *tp_as_sequence; */
 | 
						|
	&Sockinmap_as_mapping,      /* PyMappingMethods *tp_as_mapping; */
 | 
						|
 | 
						|
	/* More standard operations (here for binary compatibility) */
 | 
						|
 | 
						|
	NULL,                       /* hashfunc tp_hash; */
 | 
						|
	NULL,                       /* ternaryfunc tp_call; */
 | 
						|
	NULL,                       /* reprfunc tp_str; */
 | 
						|
	NULL,                       /* getattrofunc tp_getattro; */
 | 
						|
	NULL,                       /* setattrofunc tp_setattro; */
 | 
						|
 | 
						|
	/* Functions to access object as input/output buffer */
 | 
						|
	NULL,                       /* PyBufferProcs *tp_as_buffer; */
 | 
						|
 | 
						|
  /*** Flags to define presence of optional/expanded features ***/
 | 
						|
	Py_TPFLAGS_DEFAULT,         /* long tp_flags; */
 | 
						|
 | 
						|
	NULL,                       /*  char *tp_doc;  Documentation string */
 | 
						|
  /*** Assigned meaning in release 2.0 ***/
 | 
						|
	/* call function for all accessible objects */
 | 
						|
	NULL,                       /* traverseproc tp_traverse; */
 | 
						|
 | 
						|
	/* delete references to contained objects */
 | 
						|
	NULL,                       /* inquiry tp_clear; */
 | 
						|
 | 
						|
  /***  Assigned meaning in release 2.1 ***/
 | 
						|
  /*** rich comparisons ***/
 | 
						|
	NULL,                       /* richcmpfunc tp_richcompare; */
 | 
						|
 | 
						|
  /***  weak reference enabler ***/
 | 
						|
	0,                          /* long tp_weaklistoffset; */
 | 
						|
 | 
						|
  /*** Added in release 2.2 ***/
 | 
						|
	/*   Iterators */
 | 
						|
	0, //( getiterfunc) MVertSeq_getIter, /* getiterfunc tp_iter; */
 | 
						|
	0, //( iternextfunc ) MVertSeq_nextIter, /* iternextfunc tp_iternext; */
 | 
						|
 | 
						|
  /*** Attribute descriptor and subclassing stuff ***/
 | 
						|
	0, //BPy_MVertSeq_methods,       /* struct PyMethodDef *tp_methods; */
 | 
						|
	NULL,                       /* struct PyMemberDef *tp_members; */
 | 
						|
	NULL,                       /* struct PyGetSetDef *tp_getset; */
 | 
						|
	NULL,                       /* struct _typeobject *tp_base; */
 | 
						|
	NULL,                       /* PyObject *tp_dict; */
 | 
						|
	NULL,                       /* descrgetfunc tp_descr_get; */
 | 
						|
	NULL,                       /* descrsetfunc tp_descr_set; */
 | 
						|
	0,                          /* long tp_dictoffset; */
 | 
						|
	NULL,                       /* initproc tp_init; */
 | 
						|
	NULL,                       /* allocfunc tp_alloc; */
 | 
						|
	NULL,                       /* newfunc tp_new; */
 | 
						|
	/*  Low-level free-memory routine */
 | 
						|
	NULL,                       /* freefunc tp_free;  */
 | 
						|
	/* For PyObject_IS_GC */
 | 
						|
	NULL,                       /* inquiry tp_is_gc;  */
 | 
						|
	NULL,                       /* PyObject *tp_bases; */
 | 
						|
	/* method resolution order */
 | 
						|
	NULL,                       /* PyObject *tp_mro;  */
 | 
						|
	NULL,                       /* PyObject *tp_cache; */
 | 
						|
	NULL,                       /* PyObject *tp_subclasses; */
 | 
						|
	NULL,                       /* PyObject *tp_weaklist; */
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
static int Sockoutmap_len ( BPy_SockMap * self) {
 | 
						|
	bNode *node = self->node;
 | 
						|
	bNodeType *tinfo;
 | 
						|
	int a = 0;
 | 
						|
 | 
						|
	if (!node) return 0;
 | 
						|
 | 
						|
	tinfo = node->typeinfo;
 | 
						|
 | 
						|
	if (tinfo && tinfo->outputs) {
 | 
						|
		while(self->node->typeinfo->outputs[a].type!=-1)
 | 
						|
			a++;
 | 
						|
	}
 | 
						|
	return a;
 | 
						|
}
 | 
						|
 | 
						|
static int sockoutmap_has_key(BPy_SockMap *self, char *strkey) {
 | 
						|
	bNode *node = self->node;
 | 
						|
	bNodeType *tinfo;
 | 
						|
	int a = 0;
 | 
						|
 | 
						|
	if (!node) return -1;
 | 
						|
 | 
						|
	tinfo = node->typeinfo;
 | 
						|
 | 
						|
	if(tinfo && tinfo->outputs){
 | 
						|
		while(self->node->typeinfo->outputs[a].type!=-1) {
 | 
						|
			if(BLI_strcaseeq(self->node->typeinfo->outputs[a].name, strkey)) {
 | 
						|
				return a;
 | 
						|
			}
 | 
						|
			a++;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
static int Sockoutmap_assign_subscript(BPy_SockMap *self, PyObject *pyidx, PyObject *value) {
 | 
						|
	int i, idx, len, type, wanted_len = 0;
 | 
						|
	PyObject *val;
 | 
						|
	PyObject *items[4];
 | 
						|
 | 
						|
	if (!self->node)
 | 
						|
		return EXPP_ReturnIntError(PyExc_RuntimeError, "no access to Blender node data!");
 | 
						|
 | 
						|
	if (PyInt_Check(pyidx)) {
 | 
						|
		idx = (int)PyInt_AsLong(pyidx);
 | 
						|
		if (idx < 0 || idx >= Sockinmap_len(self))
 | 
						|
			return EXPP_ReturnIntError(PyExc_IndexError, "index out of range");
 | 
						|
	}
 | 
						|
	else if (PyString_Check(pyidx)) {
 | 
						|
		idx = sockoutmap_has_key(self, PyString_AsString(pyidx));
 | 
						|
	}
 | 
						|
	else if (PySlice_Check(pyidx)) {
 | 
						|
		return EXPP_ReturnIntError(PyExc_ValueError, "slices not yet implemented");
 | 
						|
	} else {
 | 
						|
		return EXPP_ReturnIntError(PyExc_IndexError, "index must be a positive int or a string");
 | 
						|
	}
 | 
						|
 | 
						|
	if (idx < 0)
 | 
						|
		return EXPP_ReturnIntError(PyExc_IndexError, "index must be a positive int or a string");
 | 
						|
 | 
						|
	type = self->node->typeinfo->outputs[idx].type;
 | 
						|
 | 
						|
	if (type == SOCK_VALUE) {
 | 
						|
		val = PyNumber_Float(value);
 | 
						|
		if (!val)
 | 
						|
			return EXPP_ReturnIntError(PyExc_AttributeError, "expected a float value");
 | 
						|
		self->stack[idx]->vec[0] = (float)PyFloat_AsDouble(val);
 | 
						|
		Py_DECREF(val);
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		val = PySequence_Fast(value, "expected a numeric tuple or list");
 | 
						|
		if (!val) return -1;
 | 
						|
 | 
						|
		len = PySequence_Fast_GET_SIZE(val);
 | 
						|
 | 
						|
		if (type == SOCK_VECTOR) {
 | 
						|
			wanted_len = 3;
 | 
						|
		} else { /* SOCK_RGBA */
 | 
						|
			wanted_len = 4;
 | 
						|
		}
 | 
						|
 | 
						|
		if (len != wanted_len) {
 | 
						|
			Py_DECREF(val);
 | 
						|
			PyErr_SetString(PyExc_AttributeError, "wrong number of items in list or tuple");
 | 
						|
			fprintf(stderr, "\nExpected %d numeric values, got %d.", wanted_len, len);
 | 
						|
			return -1;
 | 
						|
		}
 | 
						|
 | 
						|
		for (i = 0; i < len; i++) {
 | 
						|
			items[i] = PySequence_Fast_GET_ITEM(val, i); /* borrowed */
 | 
						|
			if (!PyNumber_Check(items[i])) {
 | 
						|
				Py_DECREF(val);
 | 
						|
				return EXPP_ReturnIntError(PyExc_AttributeError, "expected a *numeric* tuple or list");
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		self->stack[idx]->vec[0] = (float)PyFloat_AsDouble(items[0]);
 | 
						|
		self->stack[idx]->vec[1] = (float)PyFloat_AsDouble(items[1]);
 | 
						|
		self->stack[idx]->vec[2] = (float)PyFloat_AsDouble(items[2]);
 | 
						|
 | 
						|
		if (type == SOCK_RGBA)
 | 
						|
			self->stack[idx]->vec[3] = (float)PyFloat_AsDouble(items[3]);
 | 
						|
 | 
						|
		Py_DECREF(val);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int sockoutmap_set_attr(bNodeStack **stack, short type, short idx, PyObject *value)
 | 
						|
{
 | 
						|
	PyObject *val;
 | 
						|
	PyObject *items[4];
 | 
						|
	int i;
 | 
						|
	short len, wanted_len;
 | 
						|
 | 
						|
	if (type == SOCK_VALUE) {
 | 
						|
		val = PyNumber_Float(value);
 | 
						|
		if (!val)
 | 
						|
			return EXPP_ReturnIntError(PyExc_AttributeError, "expected a float value");
 | 
						|
		stack[idx]->vec[0] = (float)PyFloat_AsDouble(val);
 | 
						|
		Py_DECREF(val);
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		val = PySequence_Fast(value, "expected a numeric tuple or list");
 | 
						|
		if (!val) return -1;
 | 
						|
 | 
						|
		len = PySequence_Fast_GET_SIZE(val);
 | 
						|
 | 
						|
		if (type == SOCK_VECTOR) {
 | 
						|
			wanted_len = 3;
 | 
						|
		} else { /* SOCK_RGBA */
 | 
						|
			wanted_len = 4;
 | 
						|
		}
 | 
						|
 | 
						|
		if (len != wanted_len) {
 | 
						|
			Py_DECREF(val);
 | 
						|
			PyErr_SetString(PyExc_AttributeError, "wrong number of items in list or tuple");
 | 
						|
			fprintf(stderr, "\nExpected %d numeric values, got %d.", wanted_len, len);
 | 
						|
			return -1;
 | 
						|
		}
 | 
						|
 | 
						|
		for (i = 0; i < len; i++) {
 | 
						|
			items[i] = PySequence_Fast_GET_ITEM(val, i); /* borrowed */
 | 
						|
			if (!PyNumber_Check(items[i])) {
 | 
						|
				Py_DECREF(val);
 | 
						|
				return EXPP_ReturnIntError(PyExc_AttributeError, "expected a *numeric* tuple or list");
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		stack[idx]->vec[0] = (float)PyFloat_AsDouble(items[0]);
 | 
						|
		stack[idx]->vec[1] = (float)PyFloat_AsDouble(items[1]);
 | 
						|
		stack[idx]->vec[2] = (float)PyFloat_AsDouble(items[2]);
 | 
						|
 | 
						|
		if (type == SOCK_RGBA)
 | 
						|
			stack[idx]->vec[3] = (float)PyFloat_AsDouble(items[3]);
 | 
						|
 | 
						|
		Py_DECREF(val);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int Sockoutmap_setAttr(BPy_SockMap *self, char *name, PyObject *value) {
 | 
						|
	short idx, type;
 | 
						|
 | 
						|
	if (!self->node)
 | 
						|
		return EXPP_ReturnIntError(PyExc_RuntimeError, "no access to Blender node data!");
 | 
						|
 | 
						|
	idx = sockoutmap_has_key(self, name);
 | 
						|
 | 
						|
	if (idx < 0)
 | 
						|
		return EXPP_ReturnIntError(PyExc_AttributeError, "unknown output socket name");
 | 
						|
 | 
						|
	type = self->node->typeinfo->outputs[idx].type;
 | 
						|
 | 
						|
	return sockoutmap_set_attr(self->stack, type, idx, value);
 | 
						|
}
 | 
						|
/* write only */
 | 
						|
static PyMappingMethods Sockoutmap_as_mapping = {
 | 
						|
	( inquiry ) Sockoutmap_len,  /* mp_length */
 | 
						|
	( binaryfunc ) 0, /* mp_subscript */
 | 
						|
	( objobjargproc ) Sockoutmap_assign_subscript /* mp_ass_subscript */
 | 
						|
};
 | 
						|
 | 
						|
PyTypeObject SockOutMap_Type = {
 | 
						|
	PyObject_HEAD_INIT( NULL )  /* required py macro */
 | 
						|
	0,                          /* ob_size */
 | 
						|
	/*  For printing, in format "<module>.<name>" */
 | 
						|
	"Blender.Node.OutputSockets",           /* char *tp_name; */
 | 
						|
	sizeof( BPy_SockMap ),       /* int tp_basicsize; */
 | 
						|
	0,                          /* tp_itemsize;  For allocation */
 | 
						|
 | 
						|
	/* Methods to implement standard operations */
 | 
						|
 | 
						|
	NULL,/* destructor tp_dealloc; */
 | 
						|
	NULL,                       /* printfunc tp_print; */
 | 
						|
	NULL,                       /* getattrfunc tp_getattr; */
 | 
						|
	(setattrfunc) Sockoutmap_setAttr,/* setattrfunc tp_setattr; */
 | 
						|
	NULL,                       /* cmpfunc tp_compare; */
 | 
						|
	NULL,                       /* reprfunc tp_repr; */
 | 
						|
 | 
						|
	/* Method suites for standard classes */
 | 
						|
 | 
						|
	NULL,      					/* PyNumberMethods *tp_as_number; */
 | 
						|
	NULL,					    /* PySequenceMethods *tp_as_sequence; */
 | 
						|
	&Sockoutmap_as_mapping,      /* PyMappingMethods *tp_as_mapping; */
 | 
						|
 | 
						|
	/* More standard operations (here for binary compatibility) */
 | 
						|
 | 
						|
	NULL,                       /* hashfunc tp_hash; */
 | 
						|
	NULL,                       /* ternaryfunc tp_call; */
 | 
						|
	NULL,                       /* reprfunc tp_str; */
 | 
						|
	NULL,                       /* getattrofunc tp_getattro; */
 | 
						|
	NULL,                       /* setattrofunc tp_setattro; */
 | 
						|
 | 
						|
	/* Functions to access object as input/output buffer */
 | 
						|
	NULL,                       /* PyBufferProcs *tp_as_buffer; */
 | 
						|
 | 
						|
  /*** Flags to define presence of optional/expanded features ***/
 | 
						|
	Py_TPFLAGS_DEFAULT,         /* long tp_flags; */
 | 
						|
 | 
						|
	NULL,                       /*  char *tp_doc;  Documentation string */
 | 
						|
  /*** Assigned meaning in release 2.0 ***/
 | 
						|
	/* call function for all accessible objects */
 | 
						|
	NULL,                       /* traverseproc tp_traverse; */
 | 
						|
 | 
						|
	/* delete references to contained objects */
 | 
						|
	NULL,                       /* inquiry tp_clear; */
 | 
						|
 | 
						|
  /***  Assigned meaning in release 2.1 ***/
 | 
						|
  /*** rich comparisons ***/
 | 
						|
	NULL,                       /* richcmpfunc tp_richcompare; */
 | 
						|
 | 
						|
  /***  weak reference enabler ***/
 | 
						|
	0,                          /* long tp_weaklistoffset; */
 | 
						|
 | 
						|
  /*** Added in release 2.2 ***/
 | 
						|
	/*   Iterators */
 | 
						|
	0, //( getiterfunc) MVertSeq_getIter, /* getiterfunc tp_iter; */
 | 
						|
	0, //( iternextfunc ) MVertSeq_nextIter, /* iternextfunc tp_iternext; */
 | 
						|
 | 
						|
  /*** Attribute descriptor and subclassing stuff ***/
 | 
						|
	0, //BPy_MVertSeq_methods,       /* struct PyMethodDef *tp_methods; */
 | 
						|
	NULL,                       /* struct PyMemberDef *tp_members; */
 | 
						|
	NULL,                       /* struct PyGetSetDef *tp_getset; */
 | 
						|
	NULL,                       /* struct _typeobject *tp_base; */
 | 
						|
	NULL,                       /* PyObject *tp_dict; */
 | 
						|
	NULL,                       /* descrgetfunc tp_descr_get; */
 | 
						|
	NULL,                       /* descrsetfunc tp_descr_set; */
 | 
						|
	0,                          /* long tp_dictoffset; */
 | 
						|
	NULL,                       /* initproc tp_init; */
 | 
						|
	NULL,                       /* allocfunc tp_alloc; */
 | 
						|
	NULL,                       /* newfunc tp_new; */
 | 
						|
	/*  Low-level free-memory routine */
 | 
						|
	NULL,                       /* freefunc tp_free;  */
 | 
						|
	/* For PyObject_IS_GC */
 | 
						|
	NULL,                       /* inquiry tp_is_gc;  */
 | 
						|
	NULL,                       /* PyObject *tp_bases; */
 | 
						|
	/* method resolution order */
 | 
						|
	NULL,                       /* PyObject *tp_mro;  */
 | 
						|
	NULL,                       /* PyObject *tp_cache; */
 | 
						|
	NULL,                       /* PyObject *tp_subclasses; */
 | 
						|
	NULL,                       /* PyObject *tp_weaklist; */
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
static BPy_SockMap *Node_CreateInputMap(bNode *node, bNodeStack **stack) {
 | 
						|
	BPy_SockMap *map = PyObject_NEW(BPy_SockMap, &SockInMap_Type);
 | 
						|
	map->node = node;
 | 
						|
	map->stack = stack;
 | 
						|
	return map;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *Node_GetInputMap(BPy_Node *self) {
 | 
						|
	BPy_SockMap *inmap = Node_CreateInputMap(self->node, self->in);
 | 
						|
	return (PyObject *)(inmap);
 | 
						|
}
 | 
						|
 | 
						|
#define SURFACEVIEWVECTOR	0
 | 
						|
#define VIEWNORMAL			1
 | 
						|
#define SURFACENORMAL		2
 | 
						|
#define GLOBALTEXTURE		3
 | 
						|
#define TEXTURE				4
 | 
						|
#define PIXEL				5
 | 
						|
#define COLOR				6
 | 
						|
#define SPECULAR_COLOR		7
 | 
						|
#define MIRROR_COLOR		8
 | 
						|
#define AMBIENT_COLOR		9
 | 
						|
#define AMBIENT				10
 | 
						|
#define EMIT				11
 | 
						|
#define DISPLACE			12
 | 
						|
#define STRAND				13
 | 
						|
#define STRESS				14
 | 
						|
#define TANGENT				15
 | 
						|
#define SURFACE_D			30
 | 
						|
#define TEXTURE_D			31
 | 
						|
#define GLOBALTEXTURE_D		32
 | 
						|
#define REFLECTION_D		33
 | 
						|
#define NORMAL_D			34
 | 
						|
#define STICKY_D			35
 | 
						|
#define REFRACT_D			36
 | 
						|
#define STRAND_D			37
 | 
						|
 | 
						|
static PyObject *ShadeInput_getAttribute(BPy_ShadeInput *self, void *type) {
 | 
						|
	PyObject *obj = NULL;
 | 
						|
	if(self->shi) {
 | 
						|
		switch((int)type) {
 | 
						|
			case SURFACEVIEWVECTOR:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->view[0], self->shi->view[1], self->shi->view[2]);
 | 
						|
				break;
 | 
						|
			case VIEWNORMAL:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->vn[0], self->shi->vn[1], self->shi->vn[2]);
 | 
						|
				break;
 | 
						|
			case SURFACENORMAL:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->facenor[0], self->shi->facenor[1], self->shi->facenor[2]);
 | 
						|
				break;
 | 
						|
			case GLOBALTEXTURE:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->gl[0], self->shi->gl[1], self->shi->gl[2]);
 | 
						|
				break;
 | 
						|
			case TEXTURE:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->lo[0], self->shi->lo[1], self->shi->lo[2]);
 | 
						|
				break;
 | 
						|
			case PIXEL:
 | 
						|
				obj = Py_BuildValue("(ii)", self->shi->xs, self->shi->ys);
 | 
						|
				break;
 | 
						|
			case COLOR:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->r, self->shi->g, self->shi->b);
 | 
						|
				break;
 | 
						|
			case SPECULAR_COLOR:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->specr, self->shi->specg, self->shi->specb);
 | 
						|
				break;
 | 
						|
			case MIRROR_COLOR:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->mirr, self->shi->mirg, self->shi->mirb);
 | 
						|
				break;
 | 
						|
			case AMBIENT_COLOR:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->ambr, self->shi->ambg, self->shi->ambb);
 | 
						|
				break;
 | 
						|
			case AMBIENT:
 | 
						|
				obj = PyFloat_FromDouble((double)(self->shi->amb));
 | 
						|
				break;
 | 
						|
			case EMIT:
 | 
						|
				obj = PyFloat_FromDouble((double)(self->shi->emit));
 | 
						|
				break;
 | 
						|
			case DISPLACE:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->displace[0], self->shi->displace[1], self->shi->displace[2]);
 | 
						|
				break;
 | 
						|
			case STRAND:
 | 
						|
				obj = PyFloat_FromDouble((double)(self->shi->strandco));
 | 
						|
				break;
 | 
						|
			case STRESS:
 | 
						|
				obj = PyFloat_FromDouble((double)(self->shi->stress));
 | 
						|
				break;
 | 
						|
			case TANGENT:
 | 
						|
				obj = Py_BuildValue("(fff)", self->shi->tang[0], self->shi->tang[1], self->shi->tang[2]);
 | 
						|
				break;
 | 
						|
			case SURFACE_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxco[0], self->shi->dxco[1], self->shi->dxco[2], self->shi->dyco[0], self->shi->dyco[1], self->shi->dyco[2]);
 | 
						|
				break;
 | 
						|
			case TEXTURE_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxlo[0], self->shi->dxlo[1], self->shi->dxlo[2], self->shi->dylo[0], self->shi->dylo[1], self->shi->dylo[2]);
 | 
						|
				break;
 | 
						|
			case GLOBALTEXTURE_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxgl[0], self->shi->dxgl[1], self->shi->dxgl[2], self->shi->dygl[0], self->shi->dygl[1], self->shi->dygl[2]);
 | 
						|
				break;
 | 
						|
			case REFLECTION_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxref[0], self->shi->dxref[1], self->shi->dxref[2], self->shi->dyref[0], self->shi->dyref[1], self->shi->dyref[2]);
 | 
						|
				break;
 | 
						|
			case NORMAL_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxno[0], self->shi->dxno[1], self->shi->dxno[2], self->shi->dyno[0], self->shi->dyno[1], self->shi->dyno[2]);
 | 
						|
				break;
 | 
						|
			case STICKY_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxsticky[0], self->shi->dxsticky[1], self->shi->dxsticky[2], self->shi->dysticky[0], self->shi->dysticky[1], self->shi->dysticky[2]);
 | 
						|
				break;
 | 
						|
			case REFRACT_D:
 | 
						|
				obj = Py_BuildValue("(fff)(fff)", self->shi->dxrefract[0], self->shi->dxrefract[1], self->shi->dxrefract[2], self->shi->dyrefract[0], self->shi->dyrefract[1], self->shi->dyrefract[2]);
 | 
						|
				break;
 | 
						|
			case STRAND_D:
 | 
						|
				obj = Py_BuildValue("(ff)", self->shi->dxstrand, self->shi->dystrand);
 | 
						|
				break;
 | 
						|
			default:
 | 
						|
				break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if(!obj) {
 | 
						|
		Py_RETURN_NONE;
 | 
						|
	}
 | 
						|
	return obj;
 | 
						|
}
 | 
						|
 | 
						|
static BPy_SockMap *Node_CreateOutputMap(bNode *node, bNodeStack **stack) {
 | 
						|
	BPy_SockMap *map = PyObject_NEW(BPy_SockMap, &SockOutMap_Type);
 | 
						|
	map->node = node;
 | 
						|
	map->stack = stack;
 | 
						|
	return map;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *Node_GetOutputMap(BPy_Node *self) {
 | 
						|
	BPy_SockMap *outmap = Node_CreateOutputMap(self->node, self->out);
 | 
						|
	return (PyObject *)outmap;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *Node_GetShi(BPy_Node *self) {
 | 
						|
	BPy_ShadeInput *shi = ShadeInput_CreatePyObject(self->shi);
 | 
						|
	return (PyObject *)shi;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *node_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | 
						|
{
 | 
						|
	PyObject *self;
 | 
						|
	self = type->tp_alloc(type, 1);
 | 
						|
	return self;
 | 
						|
}
 | 
						|
 | 
						|
static int node_init(BPy_Node *self, PyObject *args, PyObject *kwds)
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static PyGetSetDef BPy_Node_getseters[] = {
 | 
						|
	{"input",
 | 
						|
		(getter)Node_GetInputMap, (setter)NULL,
 | 
						|
		"Get the input sockets mapping (dictionary)",
 | 
						|
		NULL},
 | 
						|
	{"i", /* alias */
 | 
						|
		(getter)Node_GetInputMap, (setter)NULL,
 | 
						|
		"Get the input sockets mapping (dictionary)",
 | 
						|
		NULL},
 | 
						|
	{"output",
 | 
						|
		(getter)Node_GetOutputMap, (setter)NULL,
 | 
						|
		"Get the output sockets mapping (dictionary)",
 | 
						|
		NULL},
 | 
						|
	{"o", /* alias */
 | 
						|
		(getter)Node_GetOutputMap, (setter)NULL,
 | 
						|
		"Get the output sockets mapping (dictionary)",
 | 
						|
		NULL},
 | 
						|
	{"shi",
 | 
						|
		(getter)Node_GetShi, (setter)NULL,
 | 
						|
		"Get the Shade Input data (ShadeInput)",
 | 
						|
		NULL},
 | 
						|
	{"s", /* alias */
 | 
						|
		(getter)Node_GetShi, (setter)NULL,
 | 
						|
		"Get the Shade Input data (ShadeInput)",
 | 
						|
		NULL},
 | 
						|
	{NULL,NULL,NULL,NULL,NULL}  /* Sentinel */
 | 
						|
};
 | 
						|
 | 
						|
static PyGetSetDef BPy_ShadeInput_getseters[] = {
 | 
						|
	{"texture",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the current texture coordinate (tuple)",
 | 
						|
	  (void*)TEXTURE},
 | 
						|
	{"textureGlobal",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the current global texture coordinate (tuple)",
 | 
						|
	  (void*)GLOBALTEXTURE},
 | 
						|
	{"surfaceNormal",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the current surface normal (tuple)",
 | 
						|
	  (void*)SURFACENORMAL},
 | 
						|
	{"viewNormal",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the current view normal (tuple)",
 | 
						|
	  (void*)VIEWNORMAL},
 | 
						|
	{"surfaceViewVector",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the vector pointing to the viewpoint from the point being shaded (tuple)",
 | 
						|
	  (void*)SURFACEVIEWVECTOR},
 | 
						|
	{"pixel",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the x,y-coordinate for the pixel rendered (tuple)",
 | 
						|
	  (void*)PIXEL},
 | 
						|
	{"color",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the color for the point being shaded (tuple)",
 | 
						|
	  (void*)COLOR},
 | 
						|
	{"specularColor",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the specular color for the point being shaded (tuple)",
 | 
						|
	  (void*)SPECULAR_COLOR},
 | 
						|
	{"mirrorColor",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the mirror color for the point being shaded (tuple)",
 | 
						|
	  (void*)MIRROR_COLOR},
 | 
						|
	{"ambientColor",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the ambient color for the point being shaded (tuple)",
 | 
						|
	  (void*)AMBIENT_COLOR},
 | 
						|
	{"ambient",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the ambient factor for the point being shaded (float)",
 | 
						|
	  (void*)AMBIENT},
 | 
						|
	{"emit",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the emit factor for the point being shaded (float)",
 | 
						|
	  (void*)EMIT},
 | 
						|
	{"displace",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the displace vector for the point being shaded (tuple)",
 | 
						|
	  (void*)DISPLACE},
 | 
						|
	{"strand",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the strand factor(float)",
 | 
						|
	  (void*)STRAND},
 | 
						|
	{"stress",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the stress factor(float)",
 | 
						|
	  (void*)STRESS},
 | 
						|
	{"tangent",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the tangent vector (tuple)",
 | 
						|
	  (void*)TANGENT},
 | 
						|
	{"surfaceD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the surface d (tuple of tuples)",
 | 
						|
	  (void*)SURFACE_D},
 | 
						|
	{"textureD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the texture d (tuple of tuples)",
 | 
						|
	  (void*)TEXTURE_D},
 | 
						|
	{"textureGlobalD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the global texture d (tuple of tuples)",
 | 
						|
	  (void*)GLOBALTEXTURE_D},
 | 
						|
	{"reflectionD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the reflection d (tuple of tuples)",
 | 
						|
	  (void*)REFLECTION_D},
 | 
						|
	{"normalD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the normal d (tuple of tuples)",
 | 
						|
	  (void*)NORMAL_D},
 | 
						|
	{"stickyD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the sticky d (tuple of tuples)",
 | 
						|
	  (void*)STICKY_D},
 | 
						|
	{"refractD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the refract d (tuple of tuples)",
 | 
						|
	  (void*)REFRACT_D},
 | 
						|
	{"strandD",
 | 
						|
	  (getter)ShadeInput_getAttribute, (setter)NULL,
 | 
						|
	  "Get the strand d (tuple)",
 | 
						|
	  (void*)STRAND_D},
 | 
						|
	{NULL,NULL,NULL,NULL,NULL}  /* Sentinel */
 | 
						|
};
 | 
						|
 | 
						|
PyTypeObject Node_Type = {
 | 
						|
	PyObject_HEAD_INIT( NULL )  /* required py macro */
 | 
						|
	0,                          /* ob_size */
 | 
						|
	/*  For printing, in format "<module>.<name>" */
 | 
						|
	"Blender.Node.node",             /* char *tp_name; */
 | 
						|
	sizeof( BPy_Node ),         /* int tp_basicsize; */
 | 
						|
	0,                          /* tp_itemsize;  For allocation */
 | 
						|
 | 
						|
	/* Methods to implement standard operations */
 | 
						|
 | 
						|
	NULL,/* destructor tp_dealloc; */
 | 
						|
	NULL,                       /* printfunc tp_print; */
 | 
						|
	NULL /*( getattrfunc ) PyObject_GenericGetAttr*/,                       /* getattrfunc tp_getattr; */
 | 
						|
	NULL /*( setattrfunc ) PyObject_GenericSetAttr*/,                       /* setattrfunc tp_setattr; */
 | 
						|
	( cmpfunc ) Node_compare,   /* cmpfunc tp_compare; */
 | 
						|
	( reprfunc ) Node_repr,     /* reprfunc tp_repr; */
 | 
						|
 | 
						|
	/* Method suites for standard classes */
 | 
						|
 | 
						|
	NULL,                       /* PyNumberMethods *tp_as_number; */
 | 
						|
	NULL,                       /* PySequenceMethods *tp_as_sequence; */
 | 
						|
	NULL,                       /* PyMappingMethods *tp_as_mapping; */
 | 
						|
 | 
						|
	/* More standard operations (here for binary compatibility) */
 | 
						|
 | 
						|
	NULL,                       /* hashfunc tp_hash; */
 | 
						|
	NULL,                       /* ternaryfunc tp_call; */
 | 
						|
	NULL,                       /* reprfunc tp_str; */
 | 
						|
	NULL,                       /* getattrofunc tp_getattro; */
 | 
						|
	NULL,                       /* setattrofunc tp_setattro; */
 | 
						|
 | 
						|
	/* Functions to access object as input/output buffer */
 | 
						|
	NULL,                       /* PyBufferProcs *tp_as_buffer; */
 | 
						|
 | 
						|
  /*** Flags to define presence of optional/expanded features ***/
 | 
						|
	Py_TPFLAGS_DEFAULT|Py_TPFLAGS_BASETYPE,         /* long tp_flags; */
 | 
						|
 | 
						|
	NULL,                       /*  char *tp_doc;  Documentation string */
 | 
						|
  /*** Assigned meaning in release 2.0 ***/
 | 
						|
	/* call function for all accessible objects */
 | 
						|
	NULL,                       /* traverseproc tp_traverse; */
 | 
						|
 | 
						|
	/* delete references to contained objects */
 | 
						|
	NULL,                       /* inquiry tp_clear; */
 | 
						|
 | 
						|
  /***  Assigned meaning in release 2.1 ***/
 | 
						|
  /*** rich comparisons ***/
 | 
						|
	NULL,                       /* richcmpfunc tp_richcompare; */
 | 
						|
 | 
						|
  /***  weak reference enabler ***/
 | 
						|
	0,                          /* long tp_weaklistoffset; */
 | 
						|
 | 
						|
  /*** Added in release 2.2 ***/
 | 
						|
	/*   Iterators */
 | 
						|
	NULL,                       /* getiterfunc tp_iter; */
 | 
						|
	NULL,                       /* iternextfunc tp_iternext; */
 | 
						|
 | 
						|
  /*** Attribute descriptor and subclassing stuff ***/
 | 
						|
	NULL, /*BPy_Node_methods,*/          /* struct PyMethodDef *tp_methods; */
 | 
						|
	NULL,                       /* struct PyMemberDef *tp_members; */
 | 
						|
	BPy_Node_getseters,        /* struct PyGetSetDef *tp_getset; */
 | 
						|
	NULL,                       /* struct _typeobject *tp_base; */
 | 
						|
	NULL,                       /* PyObject *tp_dict; */
 | 
						|
	NULL,                       /* descrgetfunc tp_descr_get; */
 | 
						|
	NULL,                       /* descrsetfunc tp_descr_set; */
 | 
						|
	0,                          /* long tp_dictoffset; */
 | 
						|
	(initproc)node_init,                       /* initproc tp_init; */
 | 
						|
	/*PyType_GenericAlloc*/NULL,                       /* allocfunc tp_alloc; */
 | 
						|
	node_new,                       /* newfunc tp_new; */
 | 
						|
	/*  Low-level free-memory routine */
 | 
						|
	NULL,                       /* freefunc tp_free;  */
 | 
						|
	/* For PyObject_IS_GC */
 | 
						|
	NULL,                       /* inquiry tp_is_gc;  */
 | 
						|
	NULL,                       /* PyObject *tp_bases; */
 | 
						|
	/* method resolution order */
 | 
						|
	NULL,                       /* PyObject *tp_mro;  */
 | 
						|
	NULL,                       /* PyObject *tp_cache; */
 | 
						|
	NULL,                       /* PyObject *tp_subclasses; */
 | 
						|
	NULL,                       /* PyObject *tp_weaklist; */
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
PyTypeObject ShadeInput_Type = {
 | 
						|
	PyObject_HEAD_INIT( NULL )  /* required py macro */
 | 
						|
	0,                          /* ob_size */
 | 
						|
	/*  For printing, in format "<module>.<name>" */
 | 
						|
	"Blender.Node.ShadeInput",             /* char *tp_name; */
 | 
						|
	sizeof( BPy_ShadeInput ),         /* int tp_basicsize; */
 | 
						|
	0,                          /* tp_itemsize;  For allocation */
 | 
						|
 | 
						|
	/* Methods to implement standard operations */
 | 
						|
 | 
						|
	NULL,/* destructor tp_dealloc; */
 | 
						|
	NULL,                       /* printfunc tp_print; */
 | 
						|
	NULL,                       /* getattrfunc tp_getattr; */
 | 
						|
	NULL,                       /* setattrfunc tp_setattr; */
 | 
						|
	( cmpfunc ) ShadeInput_compare,   /* cmpfunc tp_compare; */
 | 
						|
	( reprfunc ) ShadeInput_repr,     /* reprfunc tp_repr; */
 | 
						|
 | 
						|
	/* Method suites for standard classes */
 | 
						|
 | 
						|
	NULL,                       /* PyNumberMethods *tp_as_number; */
 | 
						|
	NULL,                       /* PySequenceMethods *tp_as_sequence; */
 | 
						|
	NULL,                       /* PyMappingMethods *tp_as_mapping; */
 | 
						|
 | 
						|
	/* More standard operations (here for binary compatibility) */
 | 
						|
 | 
						|
	NULL,                       /* hashfunc tp_hash; */
 | 
						|
	NULL,                       /* ternaryfunc tp_call; */
 | 
						|
	NULL,                       /* reprfunc tp_str; */
 | 
						|
	NULL,                       /* getattrofunc tp_getattro; */
 | 
						|
	NULL,                       /* setattrofunc tp_setattro; */
 | 
						|
 | 
						|
	/* Functions to access object as input/output buffer */
 | 
						|
	NULL,                       /* PyBufferProcs *tp_as_buffer; */
 | 
						|
 | 
						|
  /*** Flags to define presence of optional/expanded features ***/
 | 
						|
	Py_TPFLAGS_DEFAULT,         /* long tp_flags; */
 | 
						|
 | 
						|
	NULL,                       /*  char *tp_doc;  Documentation string */
 | 
						|
  /*** Assigned meaning in release 2.0 ***/
 | 
						|
	/* call function for all accessible objects */
 | 
						|
	NULL,                       /* traverseproc tp_traverse; */
 | 
						|
 | 
						|
	/* delete references to contained objects */
 | 
						|
	NULL,                       /* inquiry tp_clear; */
 | 
						|
 | 
						|
  /***  Assigned meaning in release 2.1 ***/
 | 
						|
  /*** rich comparisons ***/
 | 
						|
	NULL,                       /* richcmpfunc tp_richcompare; */
 | 
						|
 | 
						|
  /***  weak reference enabler ***/
 | 
						|
	0,                          /* long tp_weaklistoffset; */
 | 
						|
 | 
						|
  /*** Added in release 2.2 ***/
 | 
						|
	/*   Iterators */
 | 
						|
	NULL,                       /* getiterfunc tp_iter; */
 | 
						|
	NULL,                       /* iternextfunc tp_iternext; */
 | 
						|
 | 
						|
  /*** Attribute descriptor and subclassing stuff ***/
 | 
						|
	NULL, /*BPy_Node_methods,*/          /* struct PyMethodDef *tp_methods; */
 | 
						|
	NULL,                       /* struct PyMemberDef *tp_members; */
 | 
						|
	BPy_ShadeInput_getseters,        /* struct PyGetSetDef *tp_getset; */
 | 
						|
	NULL,                       /* struct _typeobject *tp_base; */
 | 
						|
	NULL,                       /* PyObject *tp_dict; */
 | 
						|
	NULL,                       /* descrgetfunc tp_descr_get; */
 | 
						|
	NULL,                       /* descrsetfunc tp_descr_set; */
 | 
						|
	0,                          /* long tp_dictoffset; */
 | 
						|
	NULL,                       /* initproc tp_init; */
 | 
						|
	NULL,                       /* allocfunc tp_alloc; */
 | 
						|
	NULL,                       /* newfunc tp_new; */
 | 
						|
	/*  Low-level free-memory routine */
 | 
						|
	NULL,                       /* freefunc tp_free;  */
 | 
						|
	/* For PyObject_IS_GC */
 | 
						|
	NULL,                       /* inquiry tp_is_gc;  */
 | 
						|
	NULL,                       /* PyObject *tp_bases; */
 | 
						|
	/* method resolution order */
 | 
						|
	NULL,                       /* PyObject *tp_mro;  */
 | 
						|
	NULL,                       /* PyObject *tp_cache; */
 | 
						|
	NULL,                       /* PyObject *tp_subclasses; */
 | 
						|
	NULL,                       /* PyObject *tp_weaklist; */
 | 
						|
	NULL
 | 
						|
};
 | 
						|
 | 
						|
/* Initialise Node module */
 | 
						|
PyObject *Node_Init(void)
 | 
						|
{
 | 
						|
	PyObject *submodule;
 | 
						|
 | 
						|
	if( PyType_Ready( &Node_Type ) < 0 )
 | 
						|
		return NULL;
 | 
						|
	if( PyType_Ready( &ShadeInput_Type ) < 0 )
 | 
						|
		return NULL;
 | 
						|
	if( PyType_Ready( &NodeSocket_Type ) < 0 )
 | 
						|
		return NULL;
 | 
						|
	if( PyType_Ready( &NodeSocketLists_Type ) < 0 )
 | 
						|
		return NULL;
 | 
						|
	if( PyType_Ready( &SockInMap_Type ) < 0 )
 | 
						|
		return NULL;
 | 
						|
	if( PyType_Ready( &SockOutMap_Type ) < 0 )
 | 
						|
		return NULL;
 | 
						|
	submodule = Py_InitModule3( "Blender.Node", NULL, "");
 | 
						|
 | 
						|
	PyModule_AddIntConstant(submodule, "VALUE", SOCK_VALUE);
 | 
						|
	PyModule_AddIntConstant(submodule, "RGBA", SOCK_RGBA);
 | 
						|
	PyModule_AddIntConstant(submodule, "VECTOR", SOCK_VECTOR);
 | 
						|
 | 
						|
	Py_INCREF(&NodeSocket_Type);
 | 
						|
	PyModule_AddObject(submodule, "Socket", (PyObject *)&NodeSocket_Type);
 | 
						|
 | 
						|
	Py_INCREF(&Node_Type);
 | 
						|
	PyModule_AddObject(submodule, "Scripted", (PyObject *)&Node_Type);
 | 
						|
 | 
						|
	return submodule;
 | 
						|
}
 | 
						|
 | 
						|
static int Node_compare(BPy_Node *a, BPy_Node *b)
 | 
						|
{
 | 
						|
	bNode *pa = a->node, *pb = b->node;
 | 
						|
	return (pa == pb) ? 0 : -1;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *Node_repr(BPy_Node *self)
 | 
						|
{
 | 
						|
	return PyString_FromFormat( "[Node \"%s\"]",
 | 
						|
		self->node ? self->node->id->name+2 : "empty node");
 | 
						|
}
 | 
						|
 | 
						|
BPy_Node *Node_CreatePyObject(bNode *node)
 | 
						|
{
 | 
						|
	BPy_Node *pynode;
 | 
						|
 | 
						|
	pynode = (BPy_Node *)PyObject_NEW(BPy_Node, &Node_Type);
 | 
						|
	if(!pynode) {
 | 
						|
		fprintf(stderr,"Couldn't create BPy_Node object\n");
 | 
						|
		return (BPy_Node *)(EXPP_ReturnPyObjError(PyExc_MemoryError, "couldn't create BPy_Node object"));
 | 
						|
	}
 | 
						|
 | 
						|
	pynode->node = node;
 | 
						|
 | 
						|
	return pynode;
 | 
						|
}
 | 
						|
 | 
						|
int pytype_is_pynode(PyObject *pyob)
 | 
						|
{
 | 
						|
	return PyObject_TypeCheck(pyob, &Node_Type);
 | 
						|
}
 | 
						|
 | 
						|
void InitNode(BPy_Node *self, bNode *node) {
 | 
						|
	self->node = node;
 | 
						|
}
 | 
						|
 | 
						|
bNode *Node_FromPyObject(PyObject *pyobj)
 | 
						|
{
 | 
						|
	return ((BPy_Node *)pyobj)->node;
 | 
						|
}
 | 
						|
 | 
						|
void Node_SetStack(BPy_Node *self, bNodeStack **stack, int type)
 | 
						|
{
 | 
						|
	if(type == NODE_INPUTSTACK) {
 | 
						|
		self->in = stack;
 | 
						|
	} else if(type == NODE_OUTPUTSTACK) {
 | 
						|
		self->out = stack;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Node_SetShi(BPy_Node *self, ShadeInput *shi)
 | 
						|
{
 | 
						|
	self->shi = shi;
 | 
						|
}
 | 
						|
 | 
						|
/*********************/
 | 
						|
 | 
						|
static int ShadeInput_compare(BPy_ShadeInput *a, BPy_ShadeInput *b)
 | 
						|
{
 | 
						|
	ShadeInput *pa = a->shi, *pb = b->shi;
 | 
						|
	return (pa == pb) ? 0 : -1;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *ShadeInput_repr(BPy_ShadeInput *self)
 | 
						|
{
 | 
						|
	return PyString_FromFormat( "[ShadeInput at \"%p\"]", self);
 | 
						|
}
 | 
						|
 | 
						|
BPy_ShadeInput *ShadeInput_CreatePyObject(ShadeInput *shi)
 | 
						|
{
 | 
						|
	BPy_ShadeInput *pyshi;
 | 
						|
 | 
						|
	pyshi = (BPy_ShadeInput *)PyObject_NEW(BPy_ShadeInput, &ShadeInput_Type);
 | 
						|
	if(!pyshi) {
 | 
						|
		fprintf(stderr,"Couldn't create BPy_ShadeInput object\n");
 | 
						|
		return (BPy_ShadeInput *)(EXPP_ReturnPyObjError(PyExc_MemoryError, "couldn't create BPy_ShadeInput object"));
 | 
						|
	}
 | 
						|
 | 
						|
	pyshi->shi = shi;
 | 
						|
 | 
						|
	return pyshi;
 | 
						|
}
 | 
						|
 |