4404 lines
		
	
	
		
			124 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			4404 lines
		
	
	
		
			124 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/**
 | 
						|
 * $Id$
 | 
						|
 *
 | 
						|
 * ***** BEGIN GPL 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.
 | 
						|
 *
 | 
						|
 * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 | 
						|
 *
 | 
						|
 * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
 | 
						|
 * All rights reserved.
 | 
						|
 *
 | 
						|
 * The Original Code is: all of this file.
 | 
						|
 *
 | 
						|
 * Contributor(s): 2007, Joshua Leung, major recode
 | 
						|
 *
 | 
						|
 * ***** END GPL LICENSE BLOCK *****
 | 
						|
 */
 | 
						|
 | 
						|
#include <stdio.h> 
 | 
						|
#include <stddef.h>
 | 
						|
#include <string.h>
 | 
						|
#include <math.h>
 | 
						|
#include <float.h>
 | 
						|
 | 
						|
#include "MEM_guardedalloc.h"
 | 
						|
 | 
						|
#include "BLI_blenlib.h"
 | 
						|
#include "BLI_math.h"
 | 
						|
#include "BLI_editVert.h"
 | 
						|
 | 
						|
#include "DNA_armature_types.h"
 | 
						|
#include "DNA_constraint_types.h"
 | 
						|
#include "DNA_modifier_types.h"
 | 
						|
#include "DNA_object_types.h"
 | 
						|
#include "DNA_action_types.h"
 | 
						|
#include "DNA_curve_types.h"
 | 
						|
#include "DNA_mesh_types.h"
 | 
						|
#include "DNA_meshdata_types.h"
 | 
						|
#include "DNA_lattice_types.h"
 | 
						|
#include "DNA_scene_types.h"
 | 
						|
#include "DNA_text_types.h"
 | 
						|
 | 
						|
#include "BKE_utildefines.h"
 | 
						|
#include "BKE_action.h"
 | 
						|
#include "BKE_anim.h" /* for the curve calculation part */
 | 
						|
#include "BKE_armature.h"
 | 
						|
#include "BKE_blender.h"
 | 
						|
#include "BKE_constraint.h"
 | 
						|
#include "BKE_displist.h"
 | 
						|
#include "BKE_deform.h"
 | 
						|
#include "BKE_DerivedMesh.h"	/* for geometry targets */
 | 
						|
#include "BKE_cdderivedmesh.h" /* for geometry targets */
 | 
						|
#include "BKE_object.h"
 | 
						|
#include "BKE_ipo.h"
 | 
						|
#include "BKE_global.h"
 | 
						|
#include "BKE_library.h"
 | 
						|
#include "BKE_idprop.h"
 | 
						|
#include "BKE_shrinkwrap.h"
 | 
						|
#include "BKE_mesh.h"
 | 
						|
 | 
						|
#ifndef DISABLE_PYTHON
 | 
						|
#include "BPY_extern.h"
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef HAVE_CONFIG_H
 | 
						|
#include <config.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#ifndef M_PI
 | 
						|
#define M_PI		3.14159265358979323846
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/* ************************ Constraints - General Utilities *************************** */
 | 
						|
/* These functions here don't act on any specific constraints, and are therefore should/will
 | 
						|
 * not require any of the special function-pointers afforded by the relevant constraint 
 | 
						|
 * type-info structs.
 | 
						|
 */
 | 
						|
 | 
						|
/* -------------- Naming -------------- */
 | 
						|
 | 
						|
/* Find the first available, non-duplicate name for a given constraint */
 | 
						|
void unique_constraint_name (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	BLI_uniquename(list, con, "Const", '.', offsetof(bConstraint, name), sizeof(con->name));
 | 
						|
}
 | 
						|
 | 
						|
/* ----------------- Evaluation Loop Preparation --------------- */
 | 
						|
 | 
						|
/* package an object/bone for use in constraint evaluation */
 | 
						|
/* This function MEM_calloc's a bConstraintOb struct, that will need to be freed after evaluation */
 | 
						|
bConstraintOb *constraints_make_evalob (Scene *scene, Object *ob, void *subdata, short datatype)
 | 
						|
{
 | 
						|
	bConstraintOb *cob;
 | 
						|
	
 | 
						|
	/* create regardless of whether we have any data! */
 | 
						|
	cob= MEM_callocN(sizeof(bConstraintOb), "bConstraintOb");
 | 
						|
	
 | 
						|
	/* for system time, part of deglobalization, code nicer later with local time (ton) */
 | 
						|
	cob->scene= scene;
 | 
						|
	
 | 
						|
	/* based on type of available data */
 | 
						|
	switch (datatype) {
 | 
						|
		case CONSTRAINT_OBTYPE_OBJECT:
 | 
						|
		{
 | 
						|
			/* disregard subdata... calloc should set other values right */
 | 
						|
			if (ob) {
 | 
						|
				cob->ob = ob;
 | 
						|
				cob->type = datatype;
 | 
						|
				cob->rotOrder = EULER_ORDER_DEFAULT; // TODO: when objects have rotation order too, use that
 | 
						|
				copy_m4_m4(cob->matrix, ob->obmat);
 | 
						|
			}
 | 
						|
			else
 | 
						|
				unit_m4(cob->matrix);
 | 
						|
			
 | 
						|
			copy_m4_m4(cob->startmat, cob->matrix);
 | 
						|
		}
 | 
						|
			break;
 | 
						|
		case CONSTRAINT_OBTYPE_BONE:
 | 
						|
		{
 | 
						|
			/* only set if we have valid bone, otherwise default */
 | 
						|
			if (ob && subdata) {
 | 
						|
				cob->ob = ob;
 | 
						|
				cob->pchan = (bPoseChannel *)subdata;
 | 
						|
				cob->type = datatype;
 | 
						|
				
 | 
						|
				if (cob->pchan->rotmode > 0) {
 | 
						|
					/* should be some type of Euler order */
 | 
						|
					cob->rotOrder= cob->pchan->rotmode; 
 | 
						|
				}
 | 
						|
				else {
 | 
						|
					/* Quats, so eulers should just use default order */
 | 
						|
					cob->rotOrder= EULER_ORDER_DEFAULT;
 | 
						|
				}
 | 
						|
				
 | 
						|
				/* matrix in world-space */
 | 
						|
				mul_m4_m4m4(cob->matrix, cob->pchan->pose_mat, ob->obmat);
 | 
						|
			}
 | 
						|
			else
 | 
						|
				unit_m4(cob->matrix);
 | 
						|
				
 | 
						|
			copy_m4_m4(cob->startmat, cob->matrix);
 | 
						|
		}
 | 
						|
			break;
 | 
						|
			
 | 
						|
		default: /* other types not yet handled */
 | 
						|
			unit_m4(cob->matrix);
 | 
						|
			unit_m4(cob->startmat);
 | 
						|
			break;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return cob;
 | 
						|
}
 | 
						|
 | 
						|
/* cleanup after constraint evaluation */
 | 
						|
void constraints_clear_evalob (bConstraintOb *cob)
 | 
						|
{
 | 
						|
	float delta[4][4], imat[4][4];
 | 
						|
	
 | 
						|
	/* prevent crashes */
 | 
						|
	if (cob == NULL) 
 | 
						|
		return;
 | 
						|
	
 | 
						|
	/* calculate delta of constraints evaluation */
 | 
						|
	invert_m4_m4(imat, cob->startmat);
 | 
						|
	mul_m4_m4m4(delta, imat, cob->matrix);
 | 
						|
	
 | 
						|
	/* copy matrices back to source */
 | 
						|
	switch (cob->type) {
 | 
						|
		case CONSTRAINT_OBTYPE_OBJECT:
 | 
						|
		{
 | 
						|
			/* cob->ob might not exist! */
 | 
						|
			if (cob->ob) {
 | 
						|
				/* copy new ob-matrix back to owner */
 | 
						|
				copy_m4_m4(cob->ob->obmat, cob->matrix);
 | 
						|
				
 | 
						|
				/* copy inverse of delta back to owner */
 | 
						|
				invert_m4_m4(cob->ob->constinv, delta);
 | 
						|
			}
 | 
						|
		}
 | 
						|
			break;
 | 
						|
		case CONSTRAINT_OBTYPE_BONE:
 | 
						|
		{
 | 
						|
			/* cob->ob or cob->pchan might not exist */
 | 
						|
			if (cob->ob && cob->pchan) {
 | 
						|
				/* copy new pose-matrix back to owner */
 | 
						|
				mul_m4_m4m4(cob->pchan->pose_mat, cob->matrix, cob->ob->imat);
 | 
						|
				
 | 
						|
				/* copy inverse of delta back to owner */
 | 
						|
				invert_m4_m4(cob->pchan->constinv, delta);
 | 
						|
			}
 | 
						|
		}
 | 
						|
			break;
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* free tempolary struct */
 | 
						|
	MEM_freeN(cob);
 | 
						|
}
 | 
						|
 | 
						|
/* -------------- Space-Conversion API -------------- */
 | 
						|
 | 
						|
/* This function is responsible for the correct transformations/conversions 
 | 
						|
 * of a matrix from one space to another for constraint evaluation.
 | 
						|
 * For now, this is only implemented for Objects and PoseChannels.
 | 
						|
 */
 | 
						|
void constraint_mat_convertspace (Object *ob, bPoseChannel *pchan, float mat[][4], short from, short to)
 | 
						|
{
 | 
						|
	float tempmat[4][4];
 | 
						|
	float diff_mat[4][4];
 | 
						|
	float imat[4][4];
 | 
						|
	
 | 
						|
	/* prevent crashes in these unlikely events  */
 | 
						|
	if (ob==NULL || mat==NULL) return;
 | 
						|
	/* optimise trick - check if need to do anything */
 | 
						|
	if (from == to) return;
 | 
						|
	
 | 
						|
	/* are we dealing with pose-channels or objects */
 | 
						|
	if (pchan) {
 | 
						|
		/* pose channels */
 | 
						|
		switch (from) {
 | 
						|
			case CONSTRAINT_SPACE_WORLD: /* ---------- FROM WORLDSPACE ---------- */
 | 
						|
			{
 | 
						|
				/* world to pose */
 | 
						|
				invert_m4_m4(imat, ob->obmat);
 | 
						|
				copy_m4_m4(tempmat, mat);
 | 
						|
				mul_m4_m4m4(mat, tempmat, imat);
 | 
						|
				
 | 
						|
				/* use pose-space as stepping stone for other spaces... */
 | 
						|
				if (ELEM(to, CONSTRAINT_SPACE_LOCAL, CONSTRAINT_SPACE_PARLOCAL)) {
 | 
						|
					/* call self with slightly different values */
 | 
						|
					constraint_mat_convertspace(ob, pchan, mat, CONSTRAINT_SPACE_POSE, to);
 | 
						|
				}
 | 
						|
			}
 | 
						|
				break;
 | 
						|
			case CONSTRAINT_SPACE_POSE:	/* ---------- FROM POSESPACE ---------- */
 | 
						|
			{
 | 
						|
				/* pose to world */
 | 
						|
				if (to == CONSTRAINT_SPACE_WORLD) {
 | 
						|
					copy_m4_m4(tempmat, mat);
 | 
						|
					mul_m4_m4m4(mat, tempmat, ob->obmat);
 | 
						|
				}
 | 
						|
				/* pose to local */
 | 
						|
				else if (to == CONSTRAINT_SPACE_LOCAL) {
 | 
						|
					if (pchan->bone) {
 | 
						|
						if (pchan->parent) {
 | 
						|
							float offs_bone[4][4];
 | 
						|
								
 | 
						|
							/* construct offs_bone the same way it is done in armature.c */
 | 
						|
							copy_m4_m3(offs_bone, pchan->bone->bone_mat);
 | 
						|
							VECCOPY(offs_bone[3], pchan->bone->head);
 | 
						|
							offs_bone[3][1]+= pchan->bone->parent->length;
 | 
						|
							
 | 
						|
							if (pchan->bone->flag & BONE_HINGE) {
 | 
						|
								/* pose_mat = par_pose-space_location * chan_mat */
 | 
						|
								float tmat[4][4];
 | 
						|
								
 | 
						|
								/* the rotation of the parent restposition */
 | 
						|
								copy_m4_m4(tmat, pchan->bone->parent->arm_mat);
 | 
						|
								
 | 
						|
								/* the location of actual parent transform */
 | 
						|
								VECCOPY(tmat[3], offs_bone[3]);
 | 
						|
								offs_bone[3][0]= offs_bone[3][1]= offs_bone[3][2]= 0.0f;
 | 
						|
								mul_m4_v3(pchan->parent->pose_mat, tmat[3]);
 | 
						|
								
 | 
						|
								mul_m4_m4m4(diff_mat, offs_bone, tmat);
 | 
						|
								invert_m4_m4(imat, diff_mat);
 | 
						|
							}
 | 
						|
							else {
 | 
						|
								/* pose_mat = par_pose_mat * bone_mat * chan_mat */
 | 
						|
								mul_m4_m4m4(diff_mat, offs_bone, pchan->parent->pose_mat);
 | 
						|
								invert_m4_m4(imat, diff_mat);
 | 
						|
							}
 | 
						|
						}
 | 
						|
						else {
 | 
						|
							/* pose_mat = chan_mat * arm_mat */
 | 
						|
							invert_m4_m4(imat, pchan->bone->arm_mat);
 | 
						|
						}
 | 
						|
						
 | 
						|
						copy_m4_m4(tempmat, mat);
 | 
						|
						mul_m4_m4m4(mat, tempmat, imat);
 | 
						|
					}
 | 
						|
				}
 | 
						|
				/* pose to local with parent */
 | 
						|
				else if (to == CONSTRAINT_SPACE_PARLOCAL) {
 | 
						|
					if (pchan->bone) {
 | 
						|
						invert_m4_m4(imat, pchan->bone->arm_mat);
 | 
						|
						copy_m4_m4(tempmat, mat);
 | 
						|
						mul_m4_m4m4(mat, tempmat, imat);
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
				break;
 | 
						|
			case CONSTRAINT_SPACE_LOCAL: /* ------------ FROM LOCALSPACE --------- */
 | 
						|
			{
 | 
						|
				/* local to pose - do inverse procedure that was done for pose to local */
 | 
						|
				if (pchan->bone) {
 | 
						|
					/* we need the posespace_matrix = local_matrix + (parent_posespace_matrix + restpos) */						
 | 
						|
					if (pchan->parent) {
 | 
						|
						float offs_bone[4][4];
 | 
						|
						
 | 
						|
						/* construct offs_bone the same way it is done in armature.c */
 | 
						|
						copy_m4_m3(offs_bone, pchan->bone->bone_mat);
 | 
						|
						VECCOPY(offs_bone[3], pchan->bone->head);
 | 
						|
						offs_bone[3][1]+= pchan->bone->parent->length;
 | 
						|
						
 | 
						|
						if (pchan->bone->flag & BONE_HINGE) {
 | 
						|
							/* pose_mat = par_pose-space_location * chan_mat */
 | 
						|
							float tmat[4][4];
 | 
						|
							
 | 
						|
							/* the rotation of the parent restposition */
 | 
						|
							copy_m4_m4(tmat, pchan->bone->parent->arm_mat);
 | 
						|
							
 | 
						|
							/* the location of actual parent transform */
 | 
						|
							VECCOPY(tmat[3], offs_bone[3]);
 | 
						|
							offs_bone[3][0]= offs_bone[3][1]= offs_bone[3][2]= 0.0f;
 | 
						|
							mul_m4_v3(pchan->parent->pose_mat, tmat[3]);
 | 
						|
							
 | 
						|
							mul_m4_m4m4(diff_mat, offs_bone, tmat);
 | 
						|
							copy_m4_m4(tempmat, mat);
 | 
						|
							mul_m4_m4m4(mat, tempmat, diff_mat);
 | 
						|
						}
 | 
						|
						else {
 | 
						|
							/* pose_mat = par_pose_mat * bone_mat * chan_mat */
 | 
						|
							mul_m4_m4m4(diff_mat, offs_bone, pchan->parent->pose_mat);
 | 
						|
							copy_m4_m4(tempmat, mat);
 | 
						|
							mul_m4_m4m4(mat, tempmat, diff_mat);
 | 
						|
						}
 | 
						|
					}
 | 
						|
					else {
 | 
						|
						copy_m4_m4(diff_mat, pchan->bone->arm_mat);
 | 
						|
						
 | 
						|
						copy_m4_m4(tempmat, mat);
 | 
						|
						mul_m4_m4m4(mat, tempmat, diff_mat);
 | 
						|
					}
 | 
						|
				}
 | 
						|
				
 | 
						|
				/* use pose-space as stepping stone for other spaces */
 | 
						|
				if (ELEM(to, CONSTRAINT_SPACE_WORLD, CONSTRAINT_SPACE_PARLOCAL)) {
 | 
						|
					/* call self with slightly different values */
 | 
						|
					constraint_mat_convertspace(ob, pchan, mat, CONSTRAINT_SPACE_POSE, to);
 | 
						|
				}				
 | 
						|
			}
 | 
						|
				break;
 | 
						|
			case CONSTRAINT_SPACE_PARLOCAL: /* -------------- FROM LOCAL WITH PARENT ---------- */
 | 
						|
			{
 | 
						|
				/* local + parent to pose */
 | 
						|
				if (pchan->bone) {					
 | 
						|
					copy_m4_m4(diff_mat, pchan->bone->arm_mat);
 | 
						|
					copy_m4_m4(tempmat, mat);
 | 
						|
					mul_m4_m4m4(mat, diff_mat, tempmat);
 | 
						|
				}
 | 
						|
				
 | 
						|
				/* use pose-space as stepping stone for other spaces */
 | 
						|
				if (ELEM(to, CONSTRAINT_SPACE_WORLD, CONSTRAINT_SPACE_LOCAL)) {
 | 
						|
					/* call self with slightly different values */
 | 
						|
					constraint_mat_convertspace(ob, pchan, mat, CONSTRAINT_SPACE_POSE, to);
 | 
						|
				}
 | 
						|
			}
 | 
						|
				break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		/* objects */
 | 
						|
		if (from==CONSTRAINT_SPACE_WORLD && to==CONSTRAINT_SPACE_LOCAL) {
 | 
						|
			/* check if object has a parent - otherwise this won't work */
 | 
						|
			if (ob->parent) {
 | 
						|
				/* 'subtract' parent's effects from owner */
 | 
						|
				mul_m4_m4m4(diff_mat, ob->parentinv, ob->parent->obmat);
 | 
						|
				invert_m4_m4(imat, diff_mat);
 | 
						|
				copy_m4_m4(tempmat, mat);
 | 
						|
				mul_m4_m4m4(mat, tempmat, imat);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else if (from==CONSTRAINT_SPACE_LOCAL && to==CONSTRAINT_SPACE_WORLD) {
 | 
						|
			/* check that object has a parent - otherwise this won't work */
 | 
						|
			if (ob->parent) {
 | 
						|
				/* 'add' parent's effect back to owner */
 | 
						|
				copy_m4_m4(tempmat, mat);
 | 
						|
				mul_m4_m4m4(diff_mat, ob->parentinv, ob->parent->obmat);
 | 
						|
				mul_m4_m4m4(mat, tempmat, diff_mat);
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* ------------ General Target Matrix Tools ---------- */
 | 
						|
 | 
						|
/* function that sets the given matrix based on given vertex group in mesh */
 | 
						|
static void contarget_get_mesh_mat (Scene *scene, Object *ob, char *substring, float mat[][4])
 | 
						|
{
 | 
						|
	DerivedMesh *dm = NULL;
 | 
						|
	Mesh *me= ob->data;
 | 
						|
	EditMesh *em = BKE_mesh_get_editmesh(me);
 | 
						|
	float vec[3] = {0.0f, 0.0f, 0.0f}, tvec[3];
 | 
						|
	float normal[3] = {0.0f, 0.0f, 0.0f}, plane[3];
 | 
						|
	float imat[3][3], tmat[3][3];
 | 
						|
	int dgroup;
 | 
						|
	short freeDM = 0;
 | 
						|
	
 | 
						|
	/* initialize target matrix using target matrix */
 | 
						|
	copy_m4_m4(mat, ob->obmat);
 | 
						|
	
 | 
						|
	/* get index of vertex group */
 | 
						|
	dgroup = defgroup_name_index(ob, substring);
 | 
						|
	if (dgroup < 0) return;
 | 
						|
	
 | 
						|
	/* get DerivedMesh */
 | 
						|
	if (em) {
 | 
						|
		/* target is in editmode, so get a special derived mesh */
 | 
						|
		dm = CDDM_from_editmesh(em, ob->data);
 | 
						|
		freeDM= 1;
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		/* when not in EditMode, use the 'final' derived mesh 
 | 
						|
		 *	- check if the custom data masks for derivedFinal mean that we can just use that
 | 
						|
		 *	  (this is more effficient + sufficient for most cases)
 | 
						|
		 */
 | 
						|
		if (ob->lastDataMask != CD_MASK_DERIVEDMESH) {
 | 
						|
			dm = mesh_get_derived_final(scene, ob, CD_MASK_DERIVEDMESH);
 | 
						|
			freeDM= 1;
 | 
						|
		}
 | 
						|
		else 
 | 
						|
			dm = (DerivedMesh *)ob->derivedFinal;
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* only continue if there's a valid DerivedMesh */
 | 
						|
	if (dm) {
 | 
						|
		MDeformVert *dvert = dm->getVertDataArray(dm, CD_MDEFORMVERT);
 | 
						|
		int numVerts = dm->getNumVerts(dm);
 | 
						|
		int i, j, count = 0;
 | 
						|
		float co[3], nor[3];
 | 
						|
		
 | 
						|
		/* check that dvert is a valid pointers (just in case) */
 | 
						|
		if (dvert) {
 | 
						|
			/* get the average of all verts with that are in the vertex-group */
 | 
						|
			for (i = 0; i < numVerts; i++) {	
 | 
						|
				for (j = 0; j < dvert[i].totweight; j++) {
 | 
						|
					/* does this vertex belong to nominated vertex group? */
 | 
						|
					if (dvert[i].dw[j].def_nr == dgroup) {
 | 
						|
						dm->getVertCo(dm, i, co);
 | 
						|
						dm->getVertNo(dm, i, nor);
 | 
						|
						add_v3_v3v3(vec, vec, co);
 | 
						|
						add_v3_v3v3(normal, normal, nor);
 | 
						|
						count++;
 | 
						|
						break;
 | 
						|
					}
 | 
						|
					
 | 
						|
				}
 | 
						|
			}
 | 
						|
			
 | 
						|
			
 | 
						|
			/* calculate averages of normal and coordinates */
 | 
						|
			if (count > 0) {
 | 
						|
				mul_v3_fl(vec, 1.0f / count);
 | 
						|
				mul_v3_fl(normal, 1.0f / count);
 | 
						|
			}
 | 
						|
			
 | 
						|
			
 | 
						|
			/* derive the rotation from the average normal: 
 | 
						|
			 *		- code taken from transform_manipulator.c, 
 | 
						|
			 *			calc_manipulator_stats, V3D_MANIP_NORMAL case
 | 
						|
			 */
 | 
						|
			/*	we need the transpose of the inverse for a normal... */
 | 
						|
			copy_m3_m4(imat, ob->obmat);
 | 
						|
			
 | 
						|
			invert_m3_m3(tmat, imat);
 | 
						|
			transpose_m3(tmat);
 | 
						|
			mul_m3_v3(tmat, normal);
 | 
						|
			
 | 
						|
			normalize_v3(normal);
 | 
						|
			VECCOPY(plane, tmat[1]);
 | 
						|
			
 | 
						|
			VECCOPY(tmat[2], normal);
 | 
						|
			cross_v3_v3v3(tmat[0], normal, plane);
 | 
						|
			cross_v3_v3v3(tmat[1], tmat[2], tmat[0]);
 | 
						|
			
 | 
						|
			copy_m4_m3(mat, tmat);
 | 
						|
			normalize_m4(mat);
 | 
						|
			
 | 
						|
			
 | 
						|
			/* apply the average coordinate as the new location */
 | 
						|
			mul_v3_m4v3(tvec, ob->obmat, vec);
 | 
						|
			VECCOPY(mat[3], tvec);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* free temporary DerivedMesh created (in EditMode case) */
 | 
						|
	if (dm && freeDM)
 | 
						|
		dm->release(dm);
 | 
						|
	if (em)
 | 
						|
		BKE_mesh_end_editmesh(me, em);
 | 
						|
}
 | 
						|
 | 
						|
/* function that sets the given matrix based on given vertex group in lattice */
 | 
						|
static void contarget_get_lattice_mat (Object *ob, char *substring, float mat[][4])
 | 
						|
{
 | 
						|
	Lattice *lt= (Lattice *)ob->data;
 | 
						|
	
 | 
						|
	DispList *dl = find_displist(&ob->disp, DL_VERTS);
 | 
						|
	float *co = dl?dl->verts:NULL;
 | 
						|
	BPoint *bp = lt->def;
 | 
						|
	
 | 
						|
	MDeformVert *dvert = lt->dvert;
 | 
						|
	int tot_verts= lt->pntsu*lt->pntsv*lt->pntsw;
 | 
						|
	float vec[3]= {0.0f, 0.0f, 0.0f}, tvec[3];
 | 
						|
	int dgroup=0, grouped=0;
 | 
						|
	int i, n;
 | 
						|
	
 | 
						|
	/* initialize target matrix using target matrix */
 | 
						|
	copy_m4_m4(mat, ob->obmat);
 | 
						|
	
 | 
						|
	/* get index of vertex group */
 | 
						|
	dgroup = defgroup_name_index(ob, substring);
 | 
						|
	if (dgroup < 0) return;
 | 
						|
	if (dvert == NULL) return;
 | 
						|
	
 | 
						|
	/* 1. Loop through control-points checking if in nominated vertex-group.
 | 
						|
	 * 2. If it is, add it to vec to find the average point.
 | 
						|
	 */
 | 
						|
	for (i=0; i < tot_verts; i++, dvert++) {
 | 
						|
		for (n= 0; n < dvert->totweight; n++) {
 | 
						|
			/* found match - vert is in vgroup */
 | 
						|
			if (dvert->dw[n].def_nr == dgroup) {
 | 
						|
				/* copy coordinates of point to temporary vector, then add to find average */
 | 
						|
				if (co)
 | 
						|
					memcpy(tvec, co, 3*sizeof(float));
 | 
						|
				else
 | 
						|
					memcpy(tvec, bp->vec, 3*sizeof(float));
 | 
						|
					
 | 
						|
				add_v3_v3v3(vec, vec, tvec);
 | 
						|
				grouped++;
 | 
						|
				
 | 
						|
				break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* advance pointer to coordinate data */
 | 
						|
		if (co) co+= 3;
 | 
						|
		else bp++;
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* find average location, then multiply by ob->obmat to find world-space location */
 | 
						|
	if (grouped)
 | 
						|
		mul_v3_fl(vec, 1.0f / grouped);
 | 
						|
	mul_v3_m4v3(tvec, ob->obmat, vec);
 | 
						|
	
 | 
						|
	/* copy new location to matrix */
 | 
						|
	VECCOPY(mat[3], tvec);
 | 
						|
}
 | 
						|
 | 
						|
/* generic function to get the appropriate matrix for most target cases */
 | 
						|
/* The cases where the target can be object data have not been implemented */
 | 
						|
static void constraint_target_to_mat4 (Scene *scene, Object *ob, char *substring, float mat[][4], short from, short to, float headtail)
 | 
						|
{
 | 
						|
	/*	Case OBJECT */
 | 
						|
	if (!strlen(substring)) {
 | 
						|
		copy_m4_m4(mat, ob->obmat);
 | 
						|
		constraint_mat_convertspace(ob, NULL, mat, from, to);
 | 
						|
	}
 | 
						|
	/* 	Case VERTEXGROUP */
 | 
						|
	/* Current method just takes the average location of all the points in the
 | 
						|
	 * VertexGroup, and uses that as the location value of the targets. Where 
 | 
						|
	 * possible, the orientation will also be calculated, by calculating an
 | 
						|
	 * 'average' vertex normal, and deriving the rotaation from that.
 | 
						|
	 *
 | 
						|
	 * NOTE: EditMode is not currently supported, and will most likely remain that
 | 
						|
	 *		way as constraints can only really affect things on object/bone level.
 | 
						|
	 */
 | 
						|
	else if (ob->type == OB_MESH) {
 | 
						|
		contarget_get_mesh_mat(scene, ob, substring, mat);
 | 
						|
		constraint_mat_convertspace(ob, NULL, mat, from, to);
 | 
						|
	}
 | 
						|
	else if (ob->type == OB_LATTICE) {
 | 
						|
		contarget_get_lattice_mat(ob, substring, mat);
 | 
						|
		constraint_mat_convertspace(ob, NULL, mat, from, to);
 | 
						|
	}
 | 
						|
	/*	Case BONE */
 | 
						|
	else {
 | 
						|
		bPoseChannel *pchan;
 | 
						|
		
 | 
						|
		pchan = get_pose_channel(ob->pose, substring);
 | 
						|
		if (pchan) {
 | 
						|
			/* Multiply the PoseSpace accumulation/final matrix for this
 | 
						|
			 * PoseChannel by the Armature Object's Matrix to get a worldspace
 | 
						|
			 * matrix.
 | 
						|
			 */
 | 
						|
			if (headtail < 0.000001) {
 | 
						|
				/* skip length interpolation if set to head */
 | 
						|
				mul_m4_m4m4(mat, pchan->pose_mat, ob->obmat);
 | 
						|
			}
 | 
						|
			else {
 | 
						|
				float tempmat[4][4], loc[3];
 | 
						|
				
 | 
						|
				/* interpolate along length of bone */
 | 
						|
				interp_v3_v3v3(loc, pchan->pose_head, pchan->pose_tail, headtail);	
 | 
						|
				
 | 
						|
				/* use interpolated distance for subtarget */
 | 
						|
				copy_m4_m4(tempmat, pchan->pose_mat);	
 | 
						|
				copy_v3_v3(tempmat[3], loc);
 | 
						|
				
 | 
						|
				mul_m4_m4m4(mat, tempmat, ob->obmat);
 | 
						|
			}
 | 
						|
		} 
 | 
						|
		else
 | 
						|
			copy_m4_m4(mat, ob->obmat);
 | 
						|
			
 | 
						|
		/* convert matrix space as required */
 | 
						|
		constraint_mat_convertspace(ob, pchan, mat, from, to);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* ************************* Specific Constraints ***************************** */
 | 
						|
/* Each constraint defines a set of functions, which will be called at the appropriate
 | 
						|
 * times. In addition to this, each constraint should have a type-info struct, where
 | 
						|
 * its functions are attached for use. 
 | 
						|
 */
 | 
						|
 
 | 
						|
/* Template for type-info data:
 | 
						|
 *	- make a copy of this when creating new constraints, and just change the functions
 | 
						|
 *	  pointed to as necessary
 | 
						|
 *	- although the naming of functions doesn't matter, it would help for code
 | 
						|
 *	  readability, to follow the same naming convention as is presented here
 | 
						|
 * 	- any functions that a constraint doesn't need to define, don't define
 | 
						|
 *	  for such cases, just use NULL 
 | 
						|
 *	- these should be defined after all the functions have been defined, so that
 | 
						|
 * 	  forward-definitions/prototypes don't need to be used!
 | 
						|
 *	- keep this copy #if-def'd so that future constraints can get based off this
 | 
						|
 */
 | 
						|
#if 0
 | 
						|
static bConstraintTypeInfo CTI_CONSTRNAME = {
 | 
						|
	CONSTRAINT_TYPE_CONSTRNAME, /* type */
 | 
						|
	sizeof(bConstrNameConstraint), /* size */
 | 
						|
	"ConstrName", /* name */
 | 
						|
	"bConstrNameConstraint", /* struct name */
 | 
						|
	constrname_free, /* free data */
 | 
						|
	constrname_relink, /* relink data */
 | 
						|
	constrname_id_looper, /* id looper */
 | 
						|
	constrname_copy, /* copy data */
 | 
						|
	constrname_new_data, /* new data */
 | 
						|
	constrname_get_tars, /* get constraint targets */
 | 
						|
	constrname_flush_tars, /* flush constraint targets */
 | 
						|
	constrname_get_tarmat, /* get target matrix */
 | 
						|
	constrname_evaluate /* evaluate */
 | 
						|
};
 | 
						|
#endif
 | 
						|
 | 
						|
/* This function should be used for the get_target_matrix member of all 
 | 
						|
 * constraints that are not picky about what happens to their target matrix.
 | 
						|
 */
 | 
						|
static void default_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	if (VALID_CONS_TARGET(ct))
 | 
						|
		constraint_target_to_mat4(cob->scene, ct->tar, ct->subtarget, ct->matrix, CONSTRAINT_SPACE_WORLD, ct->space, con->headtail);
 | 
						|
	else if (ct)
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
}
 | 
						|
 | 
						|
/* This following macro should be used for all standard single-target *_get_tars functions 
 | 
						|
 * to save typing and reduce maintainance woes.
 | 
						|
 * (Hopefully all compilers will be happy with the lines with just a space on them. Those are
 | 
						|
 *  really just to help this code easier to read)
 | 
						|
 */
 | 
						|
// TODO: cope with getting rotation order...
 | 
						|
#define SINGLETARGET_GET_TARS(con, datatar, datasubtarget, ct, list) \
 | 
						|
	{ \
 | 
						|
		ct= MEM_callocN(sizeof(bConstraintTarget), "tempConstraintTarget"); \
 | 
						|
		 \
 | 
						|
		ct->tar= datatar; \
 | 
						|
		strcpy(ct->subtarget, datasubtarget); \
 | 
						|
		ct->space= con->tarspace; \
 | 
						|
		ct->flag= CONSTRAINT_TAR_TEMP; \
 | 
						|
		 \
 | 
						|
		if (ct->tar) { \
 | 
						|
			if ((ct->tar->type==OB_ARMATURE) && (ct->subtarget[0])) { \
 | 
						|
				bPoseChannel *pchan= get_pose_channel(ct->tar->pose, ct->subtarget); \
 | 
						|
				ct->type = CONSTRAINT_OBTYPE_BONE; \
 | 
						|
				ct->rotOrder= (pchan) ? (pchan->rotmode) : EULER_ORDER_DEFAULT; \
 | 
						|
			}\
 | 
						|
			else if (ELEM(ct->tar->type, OB_MESH, OB_LATTICE) && (ct->subtarget[0])) { \
 | 
						|
				ct->type = CONSTRAINT_OBTYPE_VERT; \
 | 
						|
				ct->rotOrder = EULER_ORDER_DEFAULT; \
 | 
						|
			} \
 | 
						|
			else {\
 | 
						|
				ct->type = CONSTRAINT_OBTYPE_OBJECT; \
 | 
						|
				ct->rotOrder= ct->tar->rotmode; \
 | 
						|
			} \
 | 
						|
		} \
 | 
						|
		 \
 | 
						|
		BLI_addtail(list, ct); \
 | 
						|
	}
 | 
						|
	
 | 
						|
/* This following macro should be used for all standard single-target *_get_tars functions 
 | 
						|
 * to save typing and reduce maintainance woes. It does not do the subtarget related operations
 | 
						|
 * (Hopefully all compilers will be happy with the lines with just a space on them. Those are
 | 
						|
 *  really just to help this code easier to read)
 | 
						|
 */
 | 
						|
// TODO: cope with getting rotation order...
 | 
						|
#define SINGLETARGETNS_GET_TARS(con, datatar, ct, list) \
 | 
						|
	{ \
 | 
						|
		ct= MEM_callocN(sizeof(bConstraintTarget), "tempConstraintTarget"); \
 | 
						|
		 \
 | 
						|
		ct->tar= datatar; \
 | 
						|
		ct->space= con->tarspace; \
 | 
						|
		ct->flag= CONSTRAINT_TAR_TEMP; \
 | 
						|
		 \
 | 
						|
		if (ct->tar) ct->type = CONSTRAINT_OBTYPE_OBJECT; \
 | 
						|
		 \
 | 
						|
		BLI_addtail(list, ct); \
 | 
						|
	}
 | 
						|
 | 
						|
/* This following macro should be used for all standard single-target *_flush_tars functions
 | 
						|
 * to save typing and reduce maintainance woes.
 | 
						|
 * Note: the pointer to ct will be changed to point to the next in the list (as it gets removed)
 | 
						|
 * (Hopefully all compilers will be happy with the lines with just a space on them. Those are
 | 
						|
 *  really just to help this code easier to read)
 | 
						|
 */
 | 
						|
#define SINGLETARGET_FLUSH_TARS(con, datatar, datasubtarget, ct, list, nocopy) \
 | 
						|
	{ \
 | 
						|
		if (ct) { \
 | 
						|
			bConstraintTarget *ctn = ct->next; \
 | 
						|
			if (nocopy == 0) { \
 | 
						|
				datatar= ct->tar; \
 | 
						|
				strcpy(datasubtarget, ct->subtarget); \
 | 
						|
				con->tarspace= (char)ct->space; \
 | 
						|
			} \
 | 
						|
			 \
 | 
						|
			BLI_freelinkN(list, ct); \
 | 
						|
			ct= ctn; \
 | 
						|
		} \
 | 
						|
	}
 | 
						|
	
 | 
						|
/* This following macro should be used for all standard single-target *_flush_tars functions
 | 
						|
 * to save typing and reduce maintainance woes. It does not do the subtarget related operations.
 | 
						|
 * Note: the pointer to ct will be changed to point to the next in the list (as it gets removed)
 | 
						|
 * (Hopefully all compilers will be happy with the lines with just a space on them. Those are
 | 
						|
 *  really just to help this code easier to read)
 | 
						|
 */
 | 
						|
#define SINGLETARGETNS_FLUSH_TARS(con, datatar, ct, list, nocopy) \
 | 
						|
	{ \
 | 
						|
		if (ct) { \
 | 
						|
			bConstraintTarget *ctn = ct->next; \
 | 
						|
			if (nocopy == 0) { \
 | 
						|
				datatar= ct->tar; \
 | 
						|
				con->tarspace= (char)ct->space; \
 | 
						|
			} \
 | 
						|
			 \
 | 
						|
			BLI_freelinkN(list, ct); \
 | 
						|
			ct= ctn; \
 | 
						|
		} \
 | 
						|
	}
 | 
						|
 
 | 
						|
/* --------- ChildOf Constraint ------------ */
 | 
						|
 | 
						|
static void childof_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bChildOfConstraint *data= (bChildOfConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->flag = (CHILDOF_LOCX | CHILDOF_LOCY | CHILDOF_LOCZ |
 | 
						|
					CHILDOF_ROTX |CHILDOF_ROTY | CHILDOF_ROTZ |
 | 
						|
					CHILDOF_SIZEX | CHILDOF_SIZEY | CHILDOF_SIZEZ);
 | 
						|
	unit_m4(data->invmat);
 | 
						|
}
 | 
						|
 | 
						|
static void childof_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bChildOfConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int childof_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bChildOfConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void childof_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bChildOfConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void childof_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bChildOfConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float parmat[4][4], invmat[4][4], tempmat[4][4];
 | 
						|
		float loc[3], eul[3], size[3];
 | 
						|
		float loco[3], eulo[3], sizo[3];
 | 
						|
		
 | 
						|
		/* get offset (parent-inverse) matrix */
 | 
						|
		copy_m4_m4(invmat, data->invmat);
 | 
						|
		
 | 
						|
		/* extract components of both matrices */
 | 
						|
		VECCOPY(loc, ct->matrix[3]);
 | 
						|
		mat4_to_eulO( eul, ct->rotOrder,ct->matrix);
 | 
						|
		mat4_to_size( size,ct->matrix);
 | 
						|
		
 | 
						|
		VECCOPY(loco, invmat[3]);
 | 
						|
		mat4_to_eulO( eulo, cob->rotOrder,invmat);
 | 
						|
		mat4_to_size( sizo,invmat);
 | 
						|
		
 | 
						|
		/* disable channels not enabled */
 | 
						|
		if (!(data->flag & CHILDOF_LOCX)) loc[0]= loco[0]= 0.0f;
 | 
						|
		if (!(data->flag & CHILDOF_LOCY)) loc[1]= loco[1]= 0.0f;
 | 
						|
		if (!(data->flag & CHILDOF_LOCZ)) loc[2]= loco[2]= 0.0f;
 | 
						|
		if (!(data->flag & CHILDOF_ROTX)) eul[0]= eulo[0]= 0.0f;
 | 
						|
		if (!(data->flag & CHILDOF_ROTY)) eul[1]= eulo[1]= 0.0f;
 | 
						|
		if (!(data->flag & CHILDOF_ROTZ)) eul[2]= eulo[2]= 0.0f;
 | 
						|
		if (!(data->flag & CHILDOF_SIZEX)) size[0]= sizo[0]= 1.0f;
 | 
						|
		if (!(data->flag & CHILDOF_SIZEY)) size[1]= sizo[1]= 1.0f;
 | 
						|
		if (!(data->flag & CHILDOF_SIZEZ)) size[2]= sizo[2]= 1.0f;
 | 
						|
		
 | 
						|
		/* make new target mat and offset mat */
 | 
						|
		loc_eulO_size_to_mat4(ct->matrix, loc, eul, size, ct->rotOrder);
 | 
						|
		loc_eulO_size_to_mat4(invmat, loco, eulo, sizo, cob->rotOrder);
 | 
						|
		
 | 
						|
		/* multiply target (parent matrix) by offset (parent inverse) to get 
 | 
						|
		 * the effect of the parent that will be exherted on the owner
 | 
						|
		 */
 | 
						|
		mul_m4_m4m4(parmat, invmat, ct->matrix);
 | 
						|
		
 | 
						|
		/* now multiply the parent matrix by the owner matrix to get the 
 | 
						|
		 * the effect of this constraint (i.e.  owner is 'parented' to parent)
 | 
						|
		 */
 | 
						|
		copy_m4_m4(tempmat, cob->matrix);
 | 
						|
		mul_m4_m4m4(cob->matrix, tempmat, parmat); 
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_CHILDOF = {
 | 
						|
	CONSTRAINT_TYPE_CHILDOF, /* type */
 | 
						|
	sizeof(bChildOfConstraint), /* size */
 | 
						|
	"ChildOf", /* name */
 | 
						|
	"bChildOfConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	childof_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	childof_new_data, /* new data */
 | 
						|
	childof_get_tars, /* get constraint targets */
 | 
						|
	childof_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get a target matrix */
 | 
						|
	childof_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* -------- TrackTo Constraint ------- */
 | 
						|
 | 
						|
static void trackto_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bTrackToConstraint *data= (bTrackToConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->reserved1 = TRACK_Y;
 | 
						|
	data->reserved2 = UP_Z;
 | 
						|
}	
 | 
						|
 | 
						|
static void trackto_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bTrackToConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int trackto_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bTrackToConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void trackto_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bTrackToConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static int basis_cross (int n, int m)
 | 
						|
{
 | 
						|
	switch (n-m) {
 | 
						|
		case 1: 
 | 
						|
		case -2:
 | 
						|
			return 1;
 | 
						|
			
 | 
						|
		case -1: 
 | 
						|
		case 2:
 | 
						|
			return -1;
 | 
						|
			
 | 
						|
		default:
 | 
						|
			return 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void vectomat (float *vec, float *target_up, short axis, short upflag, short flags, float m[][3])
 | 
						|
{
 | 
						|
	float n[3];
 | 
						|
	float u[3]; /* vector specifying the up axis */
 | 
						|
	float proj[3];
 | 
						|
	float right[3];
 | 
						|
	float neg = -1;
 | 
						|
	int right_index;
 | 
						|
	
 | 
						|
	copy_v3_v3(n, vec);
 | 
						|
	if (normalize_v3(n) == 0.0) { 
 | 
						|
		n[0] = 0.0;
 | 
						|
		n[1] = 0.0;
 | 
						|
		n[2] = 1.0;
 | 
						|
	}
 | 
						|
	if (axis > 2) axis -= 3;
 | 
						|
	else negate_v3(n);
 | 
						|
 | 
						|
	/* n specifies the transformation of the track axis */
 | 
						|
	if (flags & TARGET_Z_UP) { 
 | 
						|
		/* target Z axis is the global up axis */
 | 
						|
		u[0] = target_up[0];
 | 
						|
		u[1] = target_up[1];
 | 
						|
		u[2] = target_up[2];
 | 
						|
	}
 | 
						|
	else { 
 | 
						|
		/* world Z axis is the global up axis */
 | 
						|
		u[0] = 0;
 | 
						|
		u[1] = 0;
 | 
						|
		u[2] = 1;
 | 
						|
	}
 | 
						|
 | 
						|
	/* project the up vector onto the plane specified by n */
 | 
						|
	project_v3_v3v3(proj, u, n); /* first u onto n... */
 | 
						|
	sub_v3_v3v3(proj, u, proj); /* then onto the plane */
 | 
						|
	/* proj specifies the transformation of the up axis */
 | 
						|
 | 
						|
	if (normalize_v3(proj) == 0.0) { /* degenerate projection */
 | 
						|
		proj[0] = 0.0;
 | 
						|
		proj[1] = 1.0;
 | 
						|
		proj[2] = 0.0;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Normalized cross product of n and proj specifies transformation of the right axis */
 | 
						|
	cross_v3_v3v3(right, proj, n);
 | 
						|
	normalize_v3(right);
 | 
						|
 | 
						|
	if (axis != upflag) {
 | 
						|
		right_index = 3 - axis - upflag;
 | 
						|
		neg = (float)basis_cross(axis, upflag);
 | 
						|
		
 | 
						|
		/* account for up direction, track direction */
 | 
						|
		m[right_index][0] = neg * right[0];
 | 
						|
		m[right_index][1] = neg * right[1];
 | 
						|
		m[right_index][2] = neg * right[2];
 | 
						|
		
 | 
						|
		m[upflag][0] = proj[0];
 | 
						|
		m[upflag][1] = proj[1];
 | 
						|
		m[upflag][2] = proj[2];
 | 
						|
		
 | 
						|
		m[axis][0] = n[0];
 | 
						|
		m[axis][1] = n[1];
 | 
						|
		m[axis][2] = n[2];
 | 
						|
	}
 | 
						|
	/* identity matrix - don't do anything if the two axes are the same */
 | 
						|
	else {
 | 
						|
		m[0][0]= m[1][1]= m[2][2]= 1.0;
 | 
						|
		m[0][1]= m[0][2]= 0.0;
 | 
						|
		m[1][0]= m[1][2]= 0.0;
 | 
						|
		m[2][0]= m[2][1]= 0.0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void trackto_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bTrackToConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float size[3], vec[3];
 | 
						|
		float totmat[3][3];
 | 
						|
		float tmat[4][4];
 | 
						|
		
 | 
						|
		/* Get size property, since ob->size is only the object's own relative size, not its global one */
 | 
						|
		mat4_to_size( size,cob->matrix);
 | 
						|
		
 | 
						|
		/* Clear the object's rotation */ 	
 | 
						|
		cob->matrix[0][0]=size[0];
 | 
						|
		cob->matrix[0][1]=0;
 | 
						|
		cob->matrix[0][2]=0;
 | 
						|
		cob->matrix[1][0]=0;
 | 
						|
		cob->matrix[1][1]=size[1];
 | 
						|
		cob->matrix[1][2]=0;
 | 
						|
		cob->matrix[2][0]=0;
 | 
						|
		cob->matrix[2][1]=0;
 | 
						|
		cob->matrix[2][2]=size[2];
 | 
						|
		
 | 
						|
		/* targetmat[2] instead of ownermat[2] is passed to vectomat
 | 
						|
		 * for backwards compatability it seems... (Aligorith)
 | 
						|
		 */
 | 
						|
		sub_v3_v3v3(vec, cob->matrix[3], ct->matrix[3]);
 | 
						|
		vectomat(vec, ct->matrix[2], 
 | 
						|
				(short)data->reserved1, (short)data->reserved2, 
 | 
						|
				data->flags, totmat);
 | 
						|
		
 | 
						|
		copy_m4_m4(tmat, cob->matrix);
 | 
						|
		mul_m4_m3m4(cob->matrix, totmat, tmat);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_TRACKTO = {
 | 
						|
	CONSTRAINT_TYPE_TRACKTO, /* type */
 | 
						|
	sizeof(bTrackToConstraint), /* size */
 | 
						|
	"TrackTo", /* name */
 | 
						|
	"bTrackToConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	trackto_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	trackto_new_data, /* new data */
 | 
						|
	trackto_get_tars, /* get constraint targets */
 | 
						|
	trackto_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	trackto_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* --------- Inverse-Kinemetics --------- */
 | 
						|
 | 
						|
static void kinematic_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bKinematicConstraint *data= (bKinematicConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->weight= (float)1.0;
 | 
						|
	data->orientweight= (float)1.0;
 | 
						|
	data->iterations = 500;
 | 
						|
	data->dist= (float)1.0;
 | 
						|
	data->flag= CONSTRAINT_IK_TIP|CONSTRAINT_IK_STRETCH|CONSTRAINT_IK_POS;
 | 
						|
}
 | 
						|
 | 
						|
static void kinematic_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bKinematicConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* chain target */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
	
 | 
						|
	/* poletarget */
 | 
						|
	func(con, (ID**)&data->poletar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int kinematic_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bKinematicConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints is used twice here */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		SINGLETARGET_GET_TARS(con, data->poletar, data->polesubtarget, ct, list)
 | 
						|
		
 | 
						|
		return 2;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void kinematic_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bKinematicConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->poletar, data->polesubtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void kinematic_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	bKinematicConstraint *data= con->data;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) 
 | 
						|
		constraint_target_to_mat4(cob->scene, ct->tar, ct->subtarget, ct->matrix, CONSTRAINT_SPACE_WORLD, ct->space, con->headtail);
 | 
						|
	else if (ct) {
 | 
						|
		if (data->flag & CONSTRAINT_IK_AUTO) {
 | 
						|
			Object *ob= cob->ob;
 | 
						|
			
 | 
						|
			if (ob == NULL) {
 | 
						|
				unit_m4(ct->matrix);
 | 
						|
			}
 | 
						|
			else {
 | 
						|
				float vec[3];
 | 
						|
				/* move grabtarget into world space */
 | 
						|
				VECCOPY(vec, data->grabtarget);
 | 
						|
				mul_m4_v3(ob->obmat, vec);
 | 
						|
				copy_m4_m4(ct->matrix, ob->obmat);
 | 
						|
				VECCOPY(ct->matrix[3], vec);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else
 | 
						|
			unit_m4(ct->matrix);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_KINEMATIC = {
 | 
						|
	CONSTRAINT_TYPE_KINEMATIC, /* type */
 | 
						|
	sizeof(bKinematicConstraint), /* size */
 | 
						|
	"IK", /* name */
 | 
						|
	"bKinematicConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	kinematic_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	kinematic_new_data, /* new data */
 | 
						|
	kinematic_get_tars, /* get constraint targets */
 | 
						|
	kinematic_flush_tars, /* flush constraint targets */
 | 
						|
	kinematic_get_tarmat, /* get target matrix */
 | 
						|
	NULL /* evaluate - solved as separate loop */
 | 
						|
};
 | 
						|
 | 
						|
/* -------- Follow-Path Constraint ---------- */
 | 
						|
 | 
						|
static void followpath_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bFollowPathConstraint *data= (bFollowPathConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->trackflag = TRACK_Y;
 | 
						|
	data->upflag = UP_Z;
 | 
						|
	data->offset = 0;
 | 
						|
	data->followflag = 0;
 | 
						|
}
 | 
						|
 | 
						|
static void followpath_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bFollowPathConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int followpath_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bFollowPathConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints without subtargets */
 | 
						|
		SINGLETARGETNS_GET_TARS(con, data->tar, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void followpath_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bFollowPathConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGETNS_FLUSH_TARS(con, data->tar, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void followpath_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	bFollowPathConstraint *data= con->data;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		Curve *cu= ct->tar->data;
 | 
						|
		float q[4], vec[4], dir[3], quat[4], radius, x1;
 | 
						|
		float totmat[4][4];
 | 
						|
		float curvetime;
 | 
						|
		
 | 
						|
		unit_m4(totmat);
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
		
 | 
						|
		/* note: when creating constraints that follow path, the curve gets the CU_PATH set now,
 | 
						|
		 *		currently for paths to work it needs to go through the bevlist/displist system (ton) 
 | 
						|
		 */
 | 
						|
		
 | 
						|
		/* only happens on reload file, but violates depsgraph still... fix! */
 | 
						|
		if (cu->path==NULL || cu->path->data==NULL)
 | 
						|
			makeDispListCurveTypes(cob->scene, ct->tar, 0);
 | 
						|
		
 | 
						|
		if (cu->path && cu->path->data) {
 | 
						|
			if ((data->followflag & FOLLOWPATH_STATIC) == 0) { 
 | 
						|
				/* animated position along curve depending on time */
 | 
						|
				if (cob->scene)
 | 
						|
					curvetime= bsystem_time(cob->scene, ct->tar, cu->ctime, 0.0) - data->offset;
 | 
						|
				else	
 | 
						|
					curvetime= cu->ctime - data->offset;
 | 
						|
				
 | 
						|
				/* ctime is now a proper var setting of Curve which gets set by Animato like any other var that's animated,
 | 
						|
				 * but this will only work if it actually is animated... 
 | 
						|
				 *
 | 
						|
				 * we divide the curvetime calculated in the previous step by the length of the path, to get a time
 | 
						|
				 * factor, which then gets clamped to lie within 0.0 - 1.0 range
 | 
						|
				 */
 | 
						|
				curvetime /= cu->pathlen;
 | 
						|
				CLAMP(curvetime, 0.0, 1.0);
 | 
						|
			}
 | 
						|
			else {
 | 
						|
				/* fixed position along curve */
 | 
						|
				curvetime= data->offset_fac;
 | 
						|
			}
 | 
						|
			
 | 
						|
			if ( where_on_path(ct->tar, curvetime, vec, dir, NULL, &radius) ) {
 | 
						|
				if (data->followflag & FOLLOWPATH_FOLLOW) {
 | 
						|
					vec_to_quat(quat, dir, (short)data->trackflag, (short)data->upflag);
 | 
						|
					
 | 
						|
					normalize_v3(dir);
 | 
						|
					q[0]= (float)cos(0.5*vec[3]);
 | 
						|
					x1= (float)sin(0.5*vec[3]);
 | 
						|
					q[1]= -x1*dir[0];
 | 
						|
					q[2]= -x1*dir[1];
 | 
						|
					q[3]= -x1*dir[2];
 | 
						|
					mul_qt_qtqt(quat, q, quat);
 | 
						|
					
 | 
						|
					quat_to_mat4(totmat, quat);
 | 
						|
				}
 | 
						|
				
 | 
						|
				if (data->followflag & FOLLOWPATH_RADIUS) {
 | 
						|
					float tmat[4][4], rmat[4][4];
 | 
						|
					scale_m4_fl(tmat, radius);
 | 
						|
					mul_m4_m4m4(rmat, totmat, tmat);
 | 
						|
					copy_m4_m4(totmat, rmat);
 | 
						|
				}
 | 
						|
				
 | 
						|
				VECCOPY(totmat[3], vec);
 | 
						|
				
 | 
						|
				mul_serie_m4(ct->matrix, ct->tar->obmat, totmat, NULL, NULL, NULL, NULL, NULL, NULL);
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	else if (ct)
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
}
 | 
						|
 | 
						|
static void followpath_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float obmat[4][4];
 | 
						|
		float size[3];
 | 
						|
		bFollowPathConstraint *data= con->data;
 | 
						|
		
 | 
						|
		/* get Object transform (loc/rot/size) to determine transformation from path */
 | 
						|
		// TODO: this used to be local at one point, but is probably more useful as-is
 | 
						|
		copy_m4_m4(obmat, cob->matrix);
 | 
						|
		
 | 
						|
		/* get scaling of object before applying constraint */
 | 
						|
		mat4_to_size(size, cob->matrix);
 | 
						|
		
 | 
						|
		/* apply targetmat - containing location on path, and rotation */
 | 
						|
		mul_serie_m4(cob->matrix, ct->matrix, obmat, NULL, NULL, NULL, NULL, NULL, NULL);
 | 
						|
		
 | 
						|
		/* un-apply scaling caused by path */
 | 
						|
		if ((data->followflag & FOLLOWPATH_RADIUS)==0) { /* XXX - assume that scale correction means that radius will have some scale error in it - Campbell */
 | 
						|
			float obsize[3];
 | 
						|
			
 | 
						|
			mat4_to_size( obsize,cob->matrix);
 | 
						|
			if (obsize[0])
 | 
						|
				mul_v3_fl(cob->matrix[0], size[0] / obsize[0]);
 | 
						|
			if (obsize[1])
 | 
						|
				mul_v3_fl(cob->matrix[1], size[1] / obsize[1]);
 | 
						|
			if (obsize[2])
 | 
						|
				mul_v3_fl(cob->matrix[2], size[2] / obsize[2]);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_FOLLOWPATH = {
 | 
						|
	CONSTRAINT_TYPE_FOLLOWPATH, /* type */
 | 
						|
	sizeof(bFollowPathConstraint), /* size */
 | 
						|
	"Follow Path", /* name */
 | 
						|
	"bFollowPathConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	followpath_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	followpath_new_data, /* new data */
 | 
						|
	followpath_get_tars, /* get constraint targets */
 | 
						|
	followpath_flush_tars, /* flush constraint targets */
 | 
						|
	followpath_get_tarmat, /* get target matrix */
 | 
						|
	followpath_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* --------- Limit Location --------- */
 | 
						|
 | 
						|
 | 
						|
static void loclimit_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bLocLimitConstraint *data = con->data;
 | 
						|
	
 | 
						|
	if (data->flag & LIMIT_XMIN) {
 | 
						|
		if (cob->matrix[3][0] < data->xmin)
 | 
						|
			cob->matrix[3][0] = data->xmin;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_XMAX) {
 | 
						|
		if (cob->matrix[3][0] > data->xmax)
 | 
						|
			cob->matrix[3][0] = data->xmax;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_YMIN) {
 | 
						|
		if (cob->matrix[3][1] < data->ymin)
 | 
						|
			cob->matrix[3][1] = data->ymin;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_YMAX) {
 | 
						|
		if (cob->matrix[3][1] > data->ymax)
 | 
						|
			cob->matrix[3][1] = data->ymax;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_ZMIN) {
 | 
						|
		if (cob->matrix[3][2] < data->zmin) 
 | 
						|
			cob->matrix[3][2] = data->zmin;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_ZMAX) {
 | 
						|
		if (cob->matrix[3][2] > data->zmax)
 | 
						|
			cob->matrix[3][2] = data->zmax;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_LOCLIMIT = {
 | 
						|
	CONSTRAINT_TYPE_LOCLIMIT, /* type */
 | 
						|
	sizeof(bLocLimitConstraint), /* size */
 | 
						|
	"Limit Location", /* name */
 | 
						|
	"bLocLimitConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	NULL, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	NULL, /* get constraint targets */
 | 
						|
	NULL, /* flush constraint targets */
 | 
						|
	NULL, /* get target matrix */
 | 
						|
	loclimit_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* -------- Limit Rotation --------- */
 | 
						|
 | 
						|
static void rotlimit_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bRotLimitConstraint *data = con->data;
 | 
						|
	float loc[3];
 | 
						|
	float eul[3];
 | 
						|
	float size[3];
 | 
						|
	
 | 
						|
	VECCOPY(loc, cob->matrix[3]);
 | 
						|
	mat4_to_size( size,cob->matrix);
 | 
						|
	
 | 
						|
	mat4_to_eulO( eul, cob->rotOrder,cob->matrix);
 | 
						|
	
 | 
						|
	/* constraint data uses radians internally */
 | 
						|
	
 | 
						|
	/* limiting of euler values... */
 | 
						|
	if (data->flag & LIMIT_XROT) {
 | 
						|
		if (eul[0] < data->xmin) 
 | 
						|
			eul[0] = data->xmin;
 | 
						|
			
 | 
						|
		if (eul[0] > data->xmax)
 | 
						|
			eul[0] = data->xmax;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_YROT) {
 | 
						|
		if (eul[1] < data->ymin)
 | 
						|
			eul[1] = data->ymin;
 | 
						|
			
 | 
						|
		if (eul[1] > data->ymax)
 | 
						|
			eul[1] = data->ymax;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_ZROT) {
 | 
						|
		if (eul[2] < data->zmin)
 | 
						|
			eul[2] = data->zmin;
 | 
						|
			
 | 
						|
		if (eul[2] > data->zmax)
 | 
						|
			eul[2] = data->zmax;
 | 
						|
	}
 | 
						|
		
 | 
						|
	loc_eulO_size_to_mat4(cob->matrix, loc, eul, size, cob->rotOrder);
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_ROTLIMIT = {
 | 
						|
	CONSTRAINT_TYPE_ROTLIMIT, /* type */
 | 
						|
	sizeof(bRotLimitConstraint), /* size */
 | 
						|
	"Limit Rotation", /* name */
 | 
						|
	"bRotLimitConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	NULL, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	NULL, /* get constraint targets */
 | 
						|
	NULL, /* flush constraint targets */
 | 
						|
	NULL, /* get target matrix */
 | 
						|
	rotlimit_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* --------- Limit Scaling --------- */
 | 
						|
 | 
						|
 | 
						|
static void sizelimit_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bSizeLimitConstraint *data = con->data;
 | 
						|
	float obsize[3], size[3];
 | 
						|
	
 | 
						|
	mat4_to_size( size,cob->matrix);
 | 
						|
	mat4_to_size( obsize,cob->matrix);
 | 
						|
	
 | 
						|
	if (data->flag & LIMIT_XMIN) {
 | 
						|
		if (size[0] < data->xmin) 
 | 
						|
			size[0] = data->xmin;	
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_XMAX) {
 | 
						|
		if (size[0] > data->xmax) 
 | 
						|
			size[0] = data->xmax;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_YMIN) {
 | 
						|
		if (size[1] < data->ymin) 
 | 
						|
			size[1] = data->ymin;	
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_YMAX) {
 | 
						|
		if (size[1] > data->ymax) 
 | 
						|
			size[1] = data->ymax;
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_ZMIN) {
 | 
						|
		if (size[2] < data->zmin) 
 | 
						|
			size[2] = data->zmin;	
 | 
						|
	}
 | 
						|
	if (data->flag & LIMIT_ZMAX) {
 | 
						|
		if (size[2] > data->zmax) 
 | 
						|
			size[2] = data->zmax;
 | 
						|
	}
 | 
						|
	
 | 
						|
	if (obsize[0]) 
 | 
						|
		mul_v3_fl(cob->matrix[0], size[0]/obsize[0]);
 | 
						|
	if (obsize[1]) 
 | 
						|
		mul_v3_fl(cob->matrix[1], size[1]/obsize[1]);
 | 
						|
	if (obsize[2]) 
 | 
						|
		mul_v3_fl(cob->matrix[2], size[2]/obsize[2]);
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_SIZELIMIT = {
 | 
						|
	CONSTRAINT_TYPE_SIZELIMIT, /* type */
 | 
						|
	sizeof(bSizeLimitConstraint), /* size */
 | 
						|
	"Limit Scaling", /* name */
 | 
						|
	"bSizeLimitConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	NULL, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	NULL, /* get constraint targets */
 | 
						|
	NULL, /* flush constraint targets */
 | 
						|
	NULL, /* get target matrix */
 | 
						|
	sizelimit_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ----------- Copy Location ------------- */
 | 
						|
 | 
						|
static void loclike_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bLocateLikeConstraint *data= (bLocateLikeConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->flag = LOCLIKE_X|LOCLIKE_Y|LOCLIKE_Z;
 | 
						|
}
 | 
						|
 | 
						|
static void loclike_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bLocateLikeConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int loclike_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bLocateLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void loclike_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bLocateLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void loclike_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bLocateLikeConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float offset[3] = {0.0f, 0.0f, 0.0f};
 | 
						|
		
 | 
						|
		if (data->flag & LOCLIKE_OFFSET)
 | 
						|
			VECCOPY(offset, cob->matrix[3]);
 | 
						|
			
 | 
						|
		if (data->flag & LOCLIKE_X) {
 | 
						|
			cob->matrix[3][0] = ct->matrix[3][0];
 | 
						|
			
 | 
						|
			if (data->flag & LOCLIKE_X_INVERT) cob->matrix[3][0] *= -1;
 | 
						|
			cob->matrix[3][0] += offset[0];
 | 
						|
		}
 | 
						|
		if (data->flag & LOCLIKE_Y) {
 | 
						|
			cob->matrix[3][1] = ct->matrix[3][1];
 | 
						|
			
 | 
						|
			if (data->flag & LOCLIKE_Y_INVERT) cob->matrix[3][1] *= -1;
 | 
						|
			cob->matrix[3][1] += offset[1];
 | 
						|
		}
 | 
						|
		if (data->flag & LOCLIKE_Z) {
 | 
						|
			cob->matrix[3][2] = ct->matrix[3][2];
 | 
						|
			
 | 
						|
			if (data->flag & LOCLIKE_Z_INVERT) cob->matrix[3][2] *= -1;
 | 
						|
			cob->matrix[3][2] += offset[2];
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_LOCLIKE = {
 | 
						|
	CONSTRAINT_TYPE_LOCLIKE, /* type */
 | 
						|
	sizeof(bLocateLikeConstraint), /* size */
 | 
						|
	"Copy Location", /* name */
 | 
						|
	"bLocateLikeConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	loclike_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	loclike_new_data, /* new data */
 | 
						|
	loclike_get_tars, /* get constraint targets */
 | 
						|
	loclike_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	loclike_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ----------- Copy Rotation ------------- */
 | 
						|
 | 
						|
static void rotlike_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bRotateLikeConstraint *data= (bRotateLikeConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->flag = ROTLIKE_X|ROTLIKE_Y|ROTLIKE_Z;
 | 
						|
}
 | 
						|
 | 
						|
static void rotlike_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bChildOfConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int rotlike_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bRotateLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void rotlike_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bRotateLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void rotlike_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bRotateLikeConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float	loc[3];
 | 
						|
		float	eul[3], obeul[3];
 | 
						|
		float	size[3];
 | 
						|
		
 | 
						|
		VECCOPY(loc, cob->matrix[3]);
 | 
						|
		mat4_to_size( size,cob->matrix);
 | 
						|
		
 | 
						|
		/* to allow compatible rotations, must get both rotations in the order of the owner... */
 | 
						|
		mat4_to_eulO( eul, cob->rotOrder,ct->matrix);
 | 
						|
		mat4_to_eulO( obeul, cob->rotOrder,cob->matrix);
 | 
						|
		
 | 
						|
		if ((data->flag & ROTLIKE_X)==0)
 | 
						|
			eul[0] = obeul[0];
 | 
						|
		else {
 | 
						|
			if (data->flag & ROTLIKE_OFFSET)
 | 
						|
				rotate_eulO(eul, cob->rotOrder, 'x', obeul[0]);
 | 
						|
			
 | 
						|
			if (data->flag & ROTLIKE_X_INVERT)
 | 
						|
				eul[0] *= -1;
 | 
						|
		}
 | 
						|
		
 | 
						|
		if ((data->flag & ROTLIKE_Y)==0)
 | 
						|
			eul[1] = obeul[1];
 | 
						|
		else {
 | 
						|
			if (data->flag & ROTLIKE_OFFSET)
 | 
						|
				rotate_eulO(eul, cob->rotOrder, 'y', obeul[1]);
 | 
						|
			
 | 
						|
			if (data->flag & ROTLIKE_Y_INVERT)
 | 
						|
				eul[1] *= -1;
 | 
						|
		}
 | 
						|
		
 | 
						|
		if ((data->flag & ROTLIKE_Z)==0)
 | 
						|
			eul[2] = obeul[2];
 | 
						|
		else {
 | 
						|
			if (data->flag & ROTLIKE_OFFSET)
 | 
						|
				rotate_eulO(eul, cob->rotOrder, 'z', obeul[2]);
 | 
						|
			
 | 
						|
			if (data->flag & ROTLIKE_Z_INVERT)
 | 
						|
				eul[2] *= -1;
 | 
						|
		}
 | 
						|
		
 | 
						|
		compatible_eul(eul, obeul);
 | 
						|
		loc_eulO_size_to_mat4(cob->matrix, loc, eul, size, cob->rotOrder);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_ROTLIKE = {
 | 
						|
	CONSTRAINT_TYPE_ROTLIKE, /* type */
 | 
						|
	sizeof(bRotateLikeConstraint), /* size */
 | 
						|
	"Copy Rotation", /* name */
 | 
						|
	"bRotateLikeConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	rotlike_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	rotlike_new_data, /* new data */
 | 
						|
	rotlike_get_tars, /* get constraint targets */
 | 
						|
	rotlike_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	rotlike_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Copy Scaling ---------- */
 | 
						|
 | 
						|
static void sizelike_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bSizeLikeConstraint *data= (bSizeLikeConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->flag = SIZELIKE_X|SIZELIKE_Y|SIZELIKE_Z;
 | 
						|
}
 | 
						|
 | 
						|
static void sizelike_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bSizeLikeConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int sizelike_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bSizeLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void sizelike_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bSizeLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void sizelike_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bSizeLikeConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float obsize[3], size[3];
 | 
						|
		
 | 
						|
		mat4_to_size(size, ct->matrix);
 | 
						|
		mat4_to_size(obsize, cob->matrix);
 | 
						|
		
 | 
						|
		if ((data->flag & SIZELIKE_X) && (obsize[0] != 0)) {
 | 
						|
			if (data->flag & SIZELIKE_OFFSET) {
 | 
						|
				size[0] += (obsize[0] - 1.0f);
 | 
						|
				mul_v3_fl(cob->matrix[0], size[0] / obsize[0]);
 | 
						|
			}
 | 
						|
			else
 | 
						|
				mul_v3_fl(cob->matrix[0], size[0] / obsize[0]);
 | 
						|
		}
 | 
						|
		if ((data->flag & SIZELIKE_Y) && (obsize[1] != 0)) {
 | 
						|
			if (data->flag & SIZELIKE_OFFSET) {
 | 
						|
				size[1] += (obsize[1] - 1.0f);
 | 
						|
				mul_v3_fl(cob->matrix[1], size[1] / obsize[1]);
 | 
						|
			}
 | 
						|
			else
 | 
						|
				mul_v3_fl(cob->matrix[1], size[1] / obsize[1]);
 | 
						|
		}
 | 
						|
		if ((data->flag & SIZELIKE_Z) && (obsize[2] != 0)) {
 | 
						|
			if (data->flag & SIZELIKE_OFFSET) {
 | 
						|
				size[2] += (obsize[2] - 1.0f);
 | 
						|
				mul_v3_fl(cob->matrix[2], size[2] / obsize[2]);
 | 
						|
			}
 | 
						|
			else
 | 
						|
				mul_v3_fl(cob->matrix[2], size[2] / obsize[2]);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_SIZELIKE = {
 | 
						|
	CONSTRAINT_TYPE_SIZELIKE, /* type */
 | 
						|
	sizeof(bSizeLikeConstraint), /* size */
 | 
						|
	"Copy Scale", /* name */
 | 
						|
	"bSizeLikeConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	sizelike_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	sizelike_new_data, /* new data */
 | 
						|
	sizelike_get_tars, /* get constraint targets */
 | 
						|
	sizelike_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	sizelike_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ----------- Copy Transforms ------------- */
 | 
						|
 | 
						|
static void translike_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bTransLikeConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int translike_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bTransLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void translike_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bTransLikeConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void translike_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		/* just copy the entire transform matrix of the target */
 | 
						|
		copy_m4_m4(cob->matrix, ct->matrix);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_TRANSLIKE = {
 | 
						|
	CONSTRAINT_TYPE_TRANSLIKE, /* type */
 | 
						|
	sizeof(bTransLikeConstraint), /* size */
 | 
						|
	"Copy Transforms", /* name */
 | 
						|
	"bTransLikeConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	translike_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	translike_get_tars, /* get constraint targets */
 | 
						|
	translike_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	translike_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Maintain Volume ---------- */
 | 
						|
 | 
						|
static void samevolume_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bSameVolumeConstraint *data= (bSameVolumeConstraint *)cdata;
 | 
						|
 | 
						|
	data->flag = SAMEVOL_Y;
 | 
						|
	data->volume = 1.0f;
 | 
						|
}
 | 
						|
 | 
						|
static void samevolume_evaluate (bConstraint *con, bConstraintOb *cob)
 | 
						|
{
 | 
						|
	bSameVolumeConstraint *data= con->data;
 | 
						|
 | 
						|
	float obsize[3];
 | 
						|
	float volume=data->volume;
 | 
						|
 | 
						|
	mat4_to_size(obsize, cob->matrix);
 | 
						|
 | 
						|
	switch (data->flag) {
 | 
						|
		case SAMEVOL_X:
 | 
						|
			if (obsize[0]!=0) {
 | 
						|
				mul_v3_fl(cob->matrix[1], sqrt(volume/obsize[0])/obsize[0]);
 | 
						|
				mul_v3_fl(cob->matrix[2], sqrt(volume/obsize[0])/obsize[0]);
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case SAMEVOL_Y:
 | 
						|
			if (obsize[1]!=0) {
 | 
						|
				mul_v3_fl(cob->matrix[0], sqrt(volume/obsize[1])/obsize[1]);
 | 
						|
				mul_v3_fl(cob->matrix[2], sqrt(volume/obsize[1])/obsize[1]);
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case SAMEVOL_Z:
 | 
						|
			if (obsize[2]!=0) {
 | 
						|
				mul_v3_fl(cob->matrix[0], sqrt(volume/obsize[2])/obsize[2]);
 | 
						|
				mul_v3_fl(cob->matrix[1], sqrt(volume/obsize[2])/obsize[2]);
 | 
						|
			}
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_SAMEVOL = {
 | 
						|
	CONSTRAINT_TYPE_SAMEVOL, /* type */
 | 
						|
	sizeof(bSameVolumeConstraint), /* size */
 | 
						|
	"Maintain Volume", /* name */
 | 
						|
	"bSameVolumeConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	NULL, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	samevolume_new_data, /* new data */
 | 
						|
	NULL, /* get constraint targets */
 | 
						|
	NULL, /* flush constraint targets */
 | 
						|
	NULL, /* get target matrix */
 | 
						|
	samevolume_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ----------- Python Constraint -------------- */
 | 
						|
 | 
						|
static void pycon_free (bConstraint *con)
 | 
						|
{
 | 
						|
	bPythonConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* id-properties */
 | 
						|
	IDP_FreeProperty(data->prop);
 | 
						|
	MEM_freeN(data->prop);
 | 
						|
	
 | 
						|
	/* multiple targets */
 | 
						|
	BLI_freelistN(&data->targets);
 | 
						|
}	
 | 
						|
 | 
						|
static void pycon_relink (bConstraint *con)
 | 
						|
{
 | 
						|
	bPythonConstraint *data= con->data;
 | 
						|
	
 | 
						|
	ID_NEW(data->text);
 | 
						|
}
 | 
						|
 | 
						|
static void pycon_copy (bConstraint *con, bConstraint *srccon)
 | 
						|
{
 | 
						|
	bPythonConstraint *pycon = (bPythonConstraint *)con->data;
 | 
						|
	bPythonConstraint *opycon = (bPythonConstraint *)srccon->data;
 | 
						|
	
 | 
						|
	pycon->prop = IDP_CopyProperty(opycon->prop);
 | 
						|
	BLI_duplicatelist(&pycon->targets, &opycon->targets);
 | 
						|
}
 | 
						|
 | 
						|
static void pycon_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bPythonConstraint *data= (bPythonConstraint *)cdata;
 | 
						|
	
 | 
						|
	/* everything should be set correctly by calloc, except for the prop->type constant.*/
 | 
						|
	data->prop = MEM_callocN(sizeof(IDProperty), "PyConstraintProps");
 | 
						|
	data->prop->type = IDP_GROUP;
 | 
						|
}
 | 
						|
 | 
						|
static int pycon_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bPythonConstraint *data= con->data;
 | 
						|
		
 | 
						|
		list->first = data->targets.first;
 | 
						|
		list->last = data->targets.last;
 | 
						|
		
 | 
						|
		return data->tarnum;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void pycon_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bPythonConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct;
 | 
						|
	
 | 
						|
	/* targets */
 | 
						|
	for (ct= data->targets.first; ct; ct= ct->next)
 | 
						|
		func(con, (ID**)&ct->tar, userdata);
 | 
						|
		
 | 
						|
	/* script */
 | 
						|
	func(con, (ID**)&data->text, userdata);
 | 
						|
}
 | 
						|
 | 
						|
/* Whether this approach is maintained remains to be seen (aligorith) */
 | 
						|
static void pycon_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	bPythonConstraint *data= con->data;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		/* special exception for curves - depsgraph issues */
 | 
						|
		if (ct->tar->type == OB_CURVE) {
 | 
						|
			Curve *cu= ct->tar->data;
 | 
						|
			
 | 
						|
			/* this check is to make sure curve objects get updated on file load correctly.*/
 | 
						|
			if (cu->path==NULL || cu->path->data==NULL) /* only happens on reload file, but violates depsgraph still... fix! */
 | 
						|
				makeDispListCurveTypes(cob->scene, ct->tar, 0);				
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* firstly calculate the matrix the normal way, then let the py-function override
 | 
						|
		 * this matrix if it needs to do so
 | 
						|
		 */
 | 
						|
		constraint_target_to_mat4(cob->scene, ct->tar, ct->subtarget, ct->matrix, CONSTRAINT_SPACE_WORLD, ct->space, con->headtail);
 | 
						|
		
 | 
						|
		/* only execute target calculation if allowed */
 | 
						|
#ifndef DISABLE_PYTHON
 | 
						|
		if (G.f & G_SCRIPT_AUTOEXEC)
 | 
						|
			BPY_pyconstraint_target(data, ct);
 | 
						|
#endif
 | 
						|
	}
 | 
						|
	else if (ct)
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
}
 | 
						|
 | 
						|
static void pycon_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
#ifdef DISABLE_PYTHON
 | 
						|
	return;
 | 
						|
#else
 | 
						|
	bPythonConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* only evaluate in python if we're allowed to do so */
 | 
						|
	if ((G.f & G_SCRIPT_AUTOEXEC)==0)  return;
 | 
						|
	
 | 
						|
/* currently removed, until I this can be re-implemented for multiple targets */
 | 
						|
#if 0
 | 
						|
	/* Firstly, run the 'driver' function which has direct access to the objects involved 
 | 
						|
	 * Technically, this is potentially dangerous as users may abuse this and cause dependency-problems,
 | 
						|
	 * but it also allows certain 'clever' rigging hacks to work.
 | 
						|
	 */
 | 
						|
	BPY_pyconstraint_driver(data, cob, targets);
 | 
						|
#endif
 | 
						|
	
 | 
						|
	/* Now, run the actual 'constraint' function, which should only access the matrices */
 | 
						|
	BPY_pyconstraint_eval(data, cob, targets);
 | 
						|
#endif /* DISABLE_PYTHON */
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_PYTHON = {
 | 
						|
	CONSTRAINT_TYPE_PYTHON, /* type */
 | 
						|
	sizeof(bPythonConstraint), /* size */
 | 
						|
	"Script", /* name */
 | 
						|
	"bPythonConstraint", /* struct name */
 | 
						|
	pycon_free, /* free data */
 | 
						|
	pycon_relink, /* relink data */
 | 
						|
	pycon_id_looper, /* id looper */
 | 
						|
	pycon_copy, /* copy data */
 | 
						|
	pycon_new_data, /* new data */
 | 
						|
	pycon_get_tars, /* get constraint targets */
 | 
						|
	NULL, /* flush constraint targets */
 | 
						|
	pycon_get_tarmat, /* get target matrix */
 | 
						|
	pycon_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* -------- Action Constraint ----------- */
 | 
						|
 | 
						|
static void actcon_relink (bConstraint *con)
 | 
						|
{
 | 
						|
	bActionConstraint *data= con->data;
 | 
						|
	ID_NEW(data->act);
 | 
						|
}
 | 
						|
 | 
						|
static void actcon_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bActionConstraint *data= (bActionConstraint *)cdata;
 | 
						|
	
 | 
						|
	/* set type to 20 (Loc X), as 0 is Rot X for backwards compatability */
 | 
						|
	data->type = 20;
 | 
						|
}
 | 
						|
 | 
						|
static void actcon_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bActionConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
	
 | 
						|
	/* action */
 | 
						|
	func(con, (ID**)&data->act, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int actcon_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bActionConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void actcon_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bActionConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void actcon_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	extern void chan_calc_mat(bPoseChannel *chan);
 | 
						|
	bActionConstraint *data = con->data;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float tempmat[4][4], vec[3];
 | 
						|
		float s, t;
 | 
						|
		short axis;
 | 
						|
		
 | 
						|
		/* initialise return matrix */
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
		
 | 
						|
		/* get the transform matrix of the target */
 | 
						|
		constraint_target_to_mat4(cob->scene, ct->tar, ct->subtarget, tempmat, CONSTRAINT_SPACE_WORLD, ct->space, con->headtail);
 | 
						|
		
 | 
						|
		/* determine where in transform range target is */
 | 
						|
		/* data->type is mapped as follows for backwards compatability:
 | 
						|
		 *	00,01,02	- rotation (it used to be like this)
 | 
						|
		 * 	10,11,12	- scaling
 | 
						|
		 *	20,21,22	- location
 | 
						|
		 */
 | 
						|
		if (data->type < 10) {
 | 
						|
			/* extract rotation (is in whatever space target should be in) */
 | 
						|
			mat4_to_eul(vec, tempmat);
 | 
						|
			vec[0] *= (float)(180.0/M_PI);
 | 
						|
			vec[1] *= (float)(180.0/M_PI);
 | 
						|
			vec[2] *= (float)(180.0/M_PI);
 | 
						|
			axis= data->type;
 | 
						|
		}
 | 
						|
		else if (data->type < 20) {
 | 
						|
			/* extract scaling (is in whatever space target should be in) */
 | 
						|
			mat4_to_size(vec, tempmat);
 | 
						|
			axis= data->type - 10;
 | 
						|
		}
 | 
						|
		else {
 | 
						|
			/* extract location */
 | 
						|
			VECCOPY(vec, tempmat[3]);
 | 
						|
			axis= data->type - 20;
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* Target defines the animation */
 | 
						|
		s = (vec[axis]-data->min) / (data->max-data->min);
 | 
						|
		CLAMP(s, 0, 1);
 | 
						|
		t = (s * (data->end-data->start)) + data->start;
 | 
						|
		
 | 
						|
		if (G.f & G_DEBUG)
 | 
						|
			printf("do Action Constraint %s - Ob %s Pchan %s \n", con->name, cob->ob->id.name+2, (cob->pchan)?cob->pchan->name:NULL);
 | 
						|
		
 | 
						|
		/* Get the appropriate information from the action */
 | 
						|
		if (cob->type == CONSTRAINT_OBTYPE_BONE) {
 | 
						|
			Object workob;
 | 
						|
			bPose *pose;
 | 
						|
			bPoseChannel *pchan, *tchan;
 | 
						|
			
 | 
						|
			/* make a temporary pose and evaluate using that */
 | 
						|
			pose = MEM_callocN(sizeof(bPose), "pose");
 | 
						|
			
 | 
						|
			/* make a copy of the bone of interest in the temp pose before evaluating action, so that it can get set 
 | 
						|
			 *	- we need to manually copy over a few settings, including rotation order, otherwise this fails
 | 
						|
			 */
 | 
						|
			pchan = cob->pchan;
 | 
						|
			
 | 
						|
			tchan= verify_pose_channel(pose, pchan->name);
 | 
						|
			tchan->rotmode= pchan->rotmode;
 | 
						|
			
 | 
						|
			/* evaluate action using workob (it will only set the PoseChannel in question) */
 | 
						|
			what_does_obaction(cob->scene, cob->ob, &workob, pose, data->act, pchan->name, t);
 | 
						|
			
 | 
						|
			/* convert animation to matrices for use here */
 | 
						|
			chan_calc_mat(tchan);
 | 
						|
			copy_m4_m4(ct->matrix, tchan->chan_mat);
 | 
						|
			
 | 
						|
			/* Clean up */
 | 
						|
			free_pose(pose);
 | 
						|
		}
 | 
						|
		else if (cob->type == CONSTRAINT_OBTYPE_OBJECT) {
 | 
						|
			Object workob;
 | 
						|
			
 | 
						|
			/* evaluate using workob */
 | 
						|
			// FIXME: we don't have any consistent standards on limiting effects on object...
 | 
						|
			what_does_obaction(cob->scene, cob->ob, &workob, NULL, data->act, NULL, t);
 | 
						|
			object_to_mat4(&workob, ct->matrix);
 | 
						|
		}
 | 
						|
		else {
 | 
						|
			/* behaviour undefined... */
 | 
						|
			puts("Error: unknown owner type for Action Constraint");
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void actcon_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float temp[4][4];
 | 
						|
		
 | 
						|
		/* Nice and simple... we just need to multiply the matrices, as the get_target_matrix
 | 
						|
		 * function has already taken care of everything else.
 | 
						|
		 */
 | 
						|
		copy_m4_m4(temp, cob->matrix);
 | 
						|
		mul_m4_m4m4(cob->matrix, ct->matrix, temp);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_ACTION = {
 | 
						|
	CONSTRAINT_TYPE_ACTION, /* type */
 | 
						|
	sizeof(bActionConstraint), /* size */
 | 
						|
	"Action", /* name */
 | 
						|
	"bActionConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	actcon_relink, /* relink data */
 | 
						|
	actcon_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	actcon_new_data, /* new data */
 | 
						|
	actcon_get_tars, /* get constraint targets */
 | 
						|
	actcon_flush_tars, /* flush constraint targets */
 | 
						|
	actcon_get_tarmat, /* get target matrix */
 | 
						|
	actcon_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* --------- Locked Track ---------- */
 | 
						|
 | 
						|
static void locktrack_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bLockTrackConstraint *data= (bLockTrackConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->trackflag = TRACK_Y;
 | 
						|
	data->lockflag = LOCK_Z;
 | 
						|
}	
 | 
						|
 | 
						|
static void locktrack_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bLockTrackConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int locktrack_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bLockTrackConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void locktrack_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bLockTrackConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void locktrack_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bLockTrackConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float vec[3],vec2[3];
 | 
						|
		float totmat[3][3];
 | 
						|
		float tmpmat[3][3];
 | 
						|
		float invmat[3][3];
 | 
						|
		float tmat[4][4];
 | 
						|
		float mdet;
 | 
						|
		
 | 
						|
		/* Vector object -> target */
 | 
						|
		sub_v3_v3v3(vec, ct->matrix[3], cob->matrix[3]);
 | 
						|
		switch (data->lockflag){
 | 
						|
		case LOCK_X: /* LOCK X */
 | 
						|
		{
 | 
						|
			switch (data->trackflag) {
 | 
						|
				case TRACK_Y: /* LOCK X TRACK Y */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[0]);
 | 
						|
					sub_v3_v3v3(totmat[1], vec, vec2);
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the x axis is fixed */
 | 
						|
					totmat[0][0] = cob->matrix[0][0];
 | 
						|
					totmat[0][1] = cob->matrix[0][1];
 | 
						|
					totmat[0][2] = cob->matrix[0][2];
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[2], totmat[0], totmat[1]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_Z: /* LOCK X TRACK Z */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[0]);
 | 
						|
					sub_v3_v3v3(totmat[2], vec, vec2);
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
					
 | 
						|
					/* the x axis is fixed */
 | 
						|
					totmat[0][0] = cob->matrix[0][0];
 | 
						|
					totmat[0][1] = cob->matrix[0][1];
 | 
						|
					totmat[0][2] = cob->matrix[0][2];
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[1], totmat[2], totmat[0]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_nY: /* LOCK X TRACK -Y */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[0]);
 | 
						|
					sub_v3_v3v3(totmat[1], vec, vec2);
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					negate_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the x axis is fixed */
 | 
						|
					totmat[0][0] = cob->matrix[0][0];
 | 
						|
					totmat[0][1] = cob->matrix[0][1];
 | 
						|
					totmat[0][2] = cob->matrix[0][2];
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[2], totmat[0], totmat[1]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_nZ: /* LOCK X TRACK -Z */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[0]);
 | 
						|
					sub_v3_v3v3(totmat[2], vec, vec2);
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
					negate_v3(totmat[2]);
 | 
						|
						
 | 
						|
					/* the x axis is fixed */
 | 
						|
					totmat[0][0] = cob->matrix[0][0];
 | 
						|
					totmat[0][1] = cob->matrix[0][1];
 | 
						|
					totmat[0][2] = cob->matrix[0][2];
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
						
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[1], totmat[2], totmat[0]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				default:
 | 
						|
				{
 | 
						|
					totmat[0][0] = 1;totmat[0][1] = 0;totmat[0][2] = 0;
 | 
						|
					totmat[1][0] = 0;totmat[1][1] = 1;totmat[1][2] = 0;
 | 
						|
					totmat[2][0] = 0;totmat[2][1] = 0;totmat[2][2] = 1;
 | 
						|
				}
 | 
						|
					break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
			break;
 | 
						|
		case LOCK_Y: /* LOCK Y */
 | 
						|
		{
 | 
						|
			switch (data->trackflag) {
 | 
						|
				case TRACK_X: /* LOCK Y TRACK X */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[1]);
 | 
						|
					sub_v3_v3v3(totmat[0], vec, vec2);
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the y axis is fixed */
 | 
						|
					totmat[1][0] = cob->matrix[1][0];
 | 
						|
					totmat[1][1] = cob->matrix[1][1];
 | 
						|
					totmat[1][2] = cob->matrix[1][2];
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[2], totmat[0], totmat[1]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_Z: /* LOCK Y TRACK Z */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[1]);
 | 
						|
					sub_v3_v3v3(totmat[2], vec, vec2);
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
					
 | 
						|
					/* the y axis is fixed */
 | 
						|
					totmat[1][0] = cob->matrix[1][0];
 | 
						|
					totmat[1][1] = cob->matrix[1][1];
 | 
						|
					totmat[1][2] = cob->matrix[1][2];
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[0], totmat[1], totmat[2]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_nX: /* LOCK Y TRACK -X */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[1]);
 | 
						|
					sub_v3_v3v3(totmat[0], vec, vec2);
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					negate_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the y axis is fixed */
 | 
						|
					totmat[1][0] = cob->matrix[1][0];
 | 
						|
					totmat[1][1] = cob->matrix[1][1];
 | 
						|
					totmat[1][2] = cob->matrix[1][2];
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[2], totmat[0], totmat[1]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_nZ: /* LOCK Y TRACK -Z */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[1]);
 | 
						|
					sub_v3_v3v3(totmat[2], vec, vec2);
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
					negate_v3(totmat[2]);
 | 
						|
					
 | 
						|
					/* the y axis is fixed */
 | 
						|
					totmat[1][0] = cob->matrix[1][0];
 | 
						|
					totmat[1][1] = cob->matrix[1][1];
 | 
						|
					totmat[1][2] = cob->matrix[1][2];
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the z axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[0], totmat[1], totmat[2]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				default:
 | 
						|
				{
 | 
						|
					totmat[0][0] = 1;totmat[0][1] = 0;totmat[0][2] = 0;
 | 
						|
					totmat[1][0] = 0;totmat[1][1] = 1;totmat[1][2] = 0;
 | 
						|
					totmat[2][0] = 0;totmat[2][1] = 0;totmat[2][2] = 1;
 | 
						|
				}
 | 
						|
					break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
			break;
 | 
						|
		case LOCK_Z: /* LOCK Z */
 | 
						|
		{
 | 
						|
			switch (data->trackflag) {
 | 
						|
				case TRACK_X: /* LOCK Z TRACK X */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[2]);
 | 
						|
					sub_v3_v3v3(totmat[0], vec, vec2);
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the z axis is fixed */
 | 
						|
					totmat[2][0] = cob->matrix[2][0];
 | 
						|
					totmat[2][1] = cob->matrix[2][1];
 | 
						|
					totmat[2][2] = cob->matrix[2][2];
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
					
 | 
						|
					/* the x axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[1], totmat[2], totmat[0]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_Y: /* LOCK Z TRACK Y */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[2]);
 | 
						|
					sub_v3_v3v3(totmat[1], vec, vec2);
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the z axis is fixed */
 | 
						|
					totmat[2][0] = cob->matrix[2][0];
 | 
						|
					totmat[2][1] = cob->matrix[2][1];
 | 
						|
					totmat[2][2] = cob->matrix[2][2];
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
						
 | 
						|
					/* the x axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[0], totmat[1], totmat[2]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_nX: /* LOCK Z TRACK -X */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[2]);
 | 
						|
					sub_v3_v3v3(totmat[0], vec, vec2);
 | 
						|
					normalize_v3(totmat[0]);
 | 
						|
					negate_v3(totmat[0]);
 | 
						|
					
 | 
						|
					/* the z axis is fixed */
 | 
						|
					totmat[2][0] = cob->matrix[2][0];
 | 
						|
					totmat[2][1] = cob->matrix[2][1];
 | 
						|
					totmat[2][2] = cob->matrix[2][2];
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
					
 | 
						|
					/* the x axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[1], totmat[2], totmat[0]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				case TRACK_nY: /* LOCK Z TRACK -Y */
 | 
						|
				{
 | 
						|
					/* Projection of Vector on the plane */
 | 
						|
					project_v3_v3v3(vec2, vec, cob->matrix[2]);
 | 
						|
					sub_v3_v3v3(totmat[1], vec, vec2);
 | 
						|
					normalize_v3(totmat[1]);
 | 
						|
					negate_v3(totmat[1]);
 | 
						|
					
 | 
						|
					/* the z axis is fixed */
 | 
						|
					totmat[2][0] = cob->matrix[2][0];
 | 
						|
					totmat[2][1] = cob->matrix[2][1];
 | 
						|
					totmat[2][2] = cob->matrix[2][2];
 | 
						|
					normalize_v3(totmat[2]);
 | 
						|
						
 | 
						|
					/* the x axis gets mapped onto a third orthogonal vector */
 | 
						|
					cross_v3_v3v3(totmat[0], totmat[1], totmat[2]);
 | 
						|
				}
 | 
						|
					break;
 | 
						|
				default:
 | 
						|
				{
 | 
						|
					totmat[0][0] = 1;totmat[0][1] = 0;totmat[0][2] = 0;
 | 
						|
					totmat[1][0] = 0;totmat[1][1] = 1;totmat[1][2] = 0;
 | 
						|
					totmat[2][0] = 0;totmat[2][1] = 0;totmat[2][2] = 1;
 | 
						|
				}
 | 
						|
					break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
		{
 | 
						|
			totmat[0][0] = 1;totmat[0][1] = 0;totmat[0][2] = 0;
 | 
						|
			totmat[1][0] = 0;totmat[1][1] = 1;totmat[1][2] = 0;
 | 
						|
			totmat[2][0] = 0;totmat[2][1] = 0;totmat[2][2] = 1;
 | 
						|
		}
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		/* Block to keep matrix heading */
 | 
						|
		tmpmat[0][0] = cob->matrix[0][0];tmpmat[0][1] = cob->matrix[0][1];tmpmat[0][2] = cob->matrix[0][2];
 | 
						|
		tmpmat[1][0] = cob->matrix[1][0];tmpmat[1][1] = cob->matrix[1][1];tmpmat[1][2] = cob->matrix[1][2];
 | 
						|
		tmpmat[2][0] = cob->matrix[2][0];tmpmat[2][1] = cob->matrix[2][1];tmpmat[2][2] = cob->matrix[2][2];
 | 
						|
		normalize_v3(tmpmat[0]);
 | 
						|
		normalize_v3(tmpmat[1]);
 | 
						|
		normalize_v3(tmpmat[2]);
 | 
						|
		invert_m3_m3(invmat, tmpmat);
 | 
						|
		mul_m3_m3m3(tmpmat, totmat, invmat);
 | 
						|
		totmat[0][0] = tmpmat[0][0];totmat[0][1] = tmpmat[0][1];totmat[0][2] = tmpmat[0][2];
 | 
						|
		totmat[1][0] = tmpmat[1][0];totmat[1][1] = tmpmat[1][1];totmat[1][2] = tmpmat[1][2];
 | 
						|
		totmat[2][0] = tmpmat[2][0];totmat[2][1] = tmpmat[2][1];totmat[2][2] = tmpmat[2][2];
 | 
						|
		
 | 
						|
		copy_m4_m4(tmat, cob->matrix);
 | 
						|
		
 | 
						|
		mdet = determinant_m3(	totmat[0][0],totmat[0][1],totmat[0][2],
 | 
						|
						totmat[1][0],totmat[1][1],totmat[1][2],
 | 
						|
						totmat[2][0],totmat[2][1],totmat[2][2]);
 | 
						|
		if (mdet==0) {
 | 
						|
			totmat[0][0] = 1;totmat[0][1] = 0;totmat[0][2] = 0;
 | 
						|
			totmat[1][0] = 0;totmat[1][1] = 1;totmat[1][2] = 0;
 | 
						|
			totmat[2][0] = 0;totmat[2][1] = 0;totmat[2][2] = 1;
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* apply out transformaton to the object */
 | 
						|
		mul_m4_m3m4(cob->matrix, totmat, tmat);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_LOCKTRACK = {
 | 
						|
	CONSTRAINT_TYPE_LOCKTRACK, /* type */
 | 
						|
	sizeof(bLockTrackConstraint), /* size */
 | 
						|
	"Locked Track", /* name */
 | 
						|
	"bLockTrackConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	locktrack_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	locktrack_new_data, /* new data */
 | 
						|
	locktrack_get_tars, /* get constraint targets */
 | 
						|
	locktrack_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	locktrack_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Limit Distance Constraint ----------- */
 | 
						|
 | 
						|
static void distlimit_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bDistLimitConstraint *data= (bDistLimitConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->dist= 0.0f;
 | 
						|
}
 | 
						|
 | 
						|
static void distlimit_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bDistLimitConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int distlimit_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bDistLimitConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void distlimit_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bDistLimitConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void distlimit_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bDistLimitConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float dvec[3], dist=0.0f, sfac=1.0f;
 | 
						|
		short clamp_surf= 0;
 | 
						|
		
 | 
						|
		/* calculate our current distance from the target */
 | 
						|
		dist= len_v3v3(cob->matrix[3], ct->matrix[3]);
 | 
						|
		
 | 
						|
		/* set distance (flag is only set when user demands it) */
 | 
						|
		if (data->dist == 0)
 | 
						|
			data->dist= dist;
 | 
						|
		
 | 
						|
		/* check if we're which way to clamp from, and calculate interpolation factor (if needed) */
 | 
						|
		if (data->mode == LIMITDIST_OUTSIDE) {
 | 
						|
			/* if inside, then move to surface */
 | 
						|
			if (dist <= data->dist) {
 | 
						|
				clamp_surf= 1;
 | 
						|
				sfac= data->dist / dist;
 | 
						|
			}
 | 
						|
			/* if soft-distance is enabled, start fading once owner is dist+softdist from the target */
 | 
						|
			else if (data->flag & LIMITDIST_USESOFT) {
 | 
						|
				if (dist <= (data->dist + data->soft)) {
 | 
						|
					
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else if (data->mode == LIMITDIST_INSIDE) {
 | 
						|
			/* if outside, then move to surface */
 | 
						|
			if (dist >= data->dist) {
 | 
						|
				clamp_surf= 1;
 | 
						|
				sfac= data->dist / dist;
 | 
						|
			}
 | 
						|
			/* if soft-distance is enabled, start fading once owner is dist-soft from the target */
 | 
						|
			else if (data->flag & LIMITDIST_USESOFT) {
 | 
						|
				// FIXME: there's a problem with "jumping" when this kicks in
 | 
						|
				if (dist >= (data->dist - data->soft)) {
 | 
						|
					sfac = (float)( data->soft*(1.0 - exp(-(dist - data->dist)/data->soft)) + data->dist );
 | 
						|
					sfac /= dist;
 | 
						|
					
 | 
						|
					clamp_surf= 1;
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else {
 | 
						|
			if (IS_EQ(dist, data->dist)==0) {
 | 
						|
				clamp_surf= 1;
 | 
						|
				sfac= data->dist / dist;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* clamp to 'surface' (i.e. move owner so that dist == data->dist) */
 | 
						|
		if (clamp_surf) {
 | 
						|
			/* simply interpolate along line formed by target -> owner */
 | 
						|
			interp_v3_v3v3(dvec, ct->matrix[3], cob->matrix[3], sfac);
 | 
						|
			
 | 
						|
			/* copy new vector onto owner */
 | 
						|
			VECCOPY(cob->matrix[3], dvec);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_DISTLIMIT = {
 | 
						|
	CONSTRAINT_TYPE_DISTLIMIT, /* type */
 | 
						|
	sizeof(bDistLimitConstraint), /* size */
 | 
						|
	"Limit Distance", /* name */
 | 
						|
	"bDistLimitConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	distlimit_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	distlimit_new_data, /* new data */
 | 
						|
	distlimit_get_tars, /* get constraint targets */
 | 
						|
	distlimit_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get a target matrix */
 | 
						|
	distlimit_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Stretch To ------------ */
 | 
						|
 | 
						|
static void stretchto_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bStretchToConstraint *data= (bStretchToConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->volmode = 0;
 | 
						|
	data->plane = 0;
 | 
						|
	data->orglength = 0.0; 
 | 
						|
	data->bulge = 1.0;
 | 
						|
}
 | 
						|
 | 
						|
static void stretchto_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bStretchToConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int stretchto_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bStretchToConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void stretchto_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bStretchToConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void stretchto_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bStretchToConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float size[3], scale[3], vec[3], xx[3], zz[3], orth[3];
 | 
						|
		float totmat[3][3];
 | 
						|
		float tmat[4][4];
 | 
						|
		float dist;
 | 
						|
		
 | 
						|
		/* store scaling before destroying obmat */
 | 
						|
		mat4_to_size( size,cob->matrix);
 | 
						|
		
 | 
						|
		/* store X orientation before destroying obmat */
 | 
						|
		xx[0] = cob->matrix[0][0];
 | 
						|
		xx[1] = cob->matrix[0][1];
 | 
						|
		xx[2] = cob->matrix[0][2];
 | 
						|
		normalize_v3(xx);
 | 
						|
		
 | 
						|
		/* store Z orientation before destroying obmat */
 | 
						|
		zz[0] = cob->matrix[2][0];
 | 
						|
		zz[1] = cob->matrix[2][1];
 | 
						|
		zz[2] = cob->matrix[2][2];
 | 
						|
		normalize_v3(zz);
 | 
						|
		
 | 
						|
		sub_v3_v3v3(vec, cob->matrix[3], ct->matrix[3]);
 | 
						|
		vec[0] /= size[0];
 | 
						|
		vec[1] /= size[1];
 | 
						|
		vec[2] /= size[2];
 | 
						|
		
 | 
						|
		dist = normalize_v3(vec);
 | 
						|
		//dist = len_v3v3( ob->obmat[3], targetmat[3]);
 | 
						|
		
 | 
						|
		/* data->orglength==0 occurs on first run, and after 'R' button is clicked */
 | 
						|
		if (data->orglength == 0)  
 | 
						|
			data->orglength = dist;
 | 
						|
		if (data->bulge == 0) 
 | 
						|
			data->bulge = 1.0;
 | 
						|
		
 | 
						|
		scale[1] = dist/data->orglength;
 | 
						|
		switch (data->volmode) {
 | 
						|
		/* volume preserving scaling */
 | 
						|
		case VOLUME_XZ :
 | 
						|
			scale[0] = 1.0f - (float)sqrt(data->bulge) + (float)sqrt(data->bulge*(data->orglength/dist));
 | 
						|
			scale[2] = scale[0];
 | 
						|
			break;
 | 
						|
		case VOLUME_X:
 | 
						|
			scale[0] = 1.0f + data->bulge * (data->orglength /dist - 1);
 | 
						|
			scale[2] = 1.0;
 | 
						|
			break;
 | 
						|
		case VOLUME_Z:
 | 
						|
			scale[0] = 1.0;
 | 
						|
			scale[2] = 1.0f + data->bulge * (data->orglength /dist - 1);
 | 
						|
			break;
 | 
						|
			/* don't care for volume */
 | 
						|
		case NO_VOLUME:
 | 
						|
			scale[0] = 1.0;
 | 
						|
			scale[2] = 1.0;
 | 
						|
			break;
 | 
						|
		default: /* should not happen, but in case*/
 | 
						|
			return;    
 | 
						|
		} /* switch (data->volmode) */
 | 
						|
 | 
						|
		/* Clear the object's rotation and scale */
 | 
						|
		cob->matrix[0][0]=size[0]*scale[0];
 | 
						|
		cob->matrix[0][1]=0;
 | 
						|
		cob->matrix[0][2]=0;
 | 
						|
		cob->matrix[1][0]=0;
 | 
						|
		cob->matrix[1][1]=size[1]*scale[1];
 | 
						|
		cob->matrix[1][2]=0;
 | 
						|
		cob->matrix[2][0]=0;
 | 
						|
		cob->matrix[2][1]=0;
 | 
						|
		cob->matrix[2][2]=size[2]*scale[2];
 | 
						|
		
 | 
						|
		sub_v3_v3v3(vec, cob->matrix[3], ct->matrix[3]);
 | 
						|
		normalize_v3(vec);
 | 
						|
		
 | 
						|
		/* new Y aligns  object target connection*/
 | 
						|
		totmat[1][0] = -vec[0];
 | 
						|
		totmat[1][1] = -vec[1];
 | 
						|
		totmat[1][2] = -vec[2];
 | 
						|
		switch (data->plane) {
 | 
						|
		case PLANE_X:
 | 
						|
			/* build new Z vector */
 | 
						|
			/* othogonal to "new Y" "old X! plane */
 | 
						|
			cross_v3_v3v3(orth, vec, xx);
 | 
						|
			normalize_v3(orth);
 | 
						|
			
 | 
						|
			/* new Z*/
 | 
						|
			totmat[2][0] = orth[0];
 | 
						|
			totmat[2][1] = orth[1];
 | 
						|
			totmat[2][2] = orth[2];
 | 
						|
			
 | 
						|
			/* we decided to keep X plane*/
 | 
						|
			cross_v3_v3v3(xx, orth, vec);
 | 
						|
			normalize_v3(xx);
 | 
						|
			totmat[0][0] = xx[0];
 | 
						|
			totmat[0][1] = xx[1];
 | 
						|
			totmat[0][2] = xx[2];
 | 
						|
			break;
 | 
						|
		case PLANE_Z:
 | 
						|
			/* build new X vector */
 | 
						|
			/* othogonal to "new Y" "old Z! plane */
 | 
						|
			cross_v3_v3v3(orth, vec, zz);
 | 
						|
			normalize_v3(orth);
 | 
						|
			
 | 
						|
			/* new X */
 | 
						|
			totmat[0][0] = -orth[0];
 | 
						|
			totmat[0][1] = -orth[1];
 | 
						|
			totmat[0][2] = -orth[2];
 | 
						|
			
 | 
						|
			/* we decided to keep Z */
 | 
						|
			cross_v3_v3v3(zz, orth, vec);
 | 
						|
			normalize_v3(zz);
 | 
						|
			totmat[2][0] = zz[0];
 | 
						|
			totmat[2][1] = zz[1];
 | 
						|
			totmat[2][2] = zz[2];
 | 
						|
			break;
 | 
						|
		} /* switch (data->plane) */
 | 
						|
		
 | 
						|
		copy_m4_m4(tmat, cob->matrix);
 | 
						|
		mul_m4_m3m4(cob->matrix, totmat, tmat);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_STRETCHTO = {
 | 
						|
	CONSTRAINT_TYPE_STRETCHTO, /* type */
 | 
						|
	sizeof(bStretchToConstraint), /* size */
 | 
						|
	"Stretch To", /* name */
 | 
						|
	"bStretchToConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	stretchto_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	stretchto_new_data, /* new data */
 | 
						|
	stretchto_get_tars, /* get constraint targets */
 | 
						|
	stretchto_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	stretchto_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Floor ------------ */
 | 
						|
 | 
						|
static void minmax_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bMinMaxConstraint *data= (bMinMaxConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->minmaxflag = TRACK_Z;
 | 
						|
	data->offset = 0.0f;
 | 
						|
	data->cache[0] = data->cache[1] = data->cache[2] = 0.0f;
 | 
						|
	data->flag = 0;
 | 
						|
}
 | 
						|
 | 
						|
static void minmax_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bMinMaxConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int minmax_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bMinMaxConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void minmax_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bMinMaxConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void minmax_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bMinMaxConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float obmat[4][4], imat[4][4], tarmat[4][4], tmat[4][4];
 | 
						|
		float val1, val2;
 | 
						|
		int index;
 | 
						|
		
 | 
						|
		copy_m4_m4(obmat, cob->matrix);
 | 
						|
		copy_m4_m4(tarmat, ct->matrix);
 | 
						|
		
 | 
						|
		if (data->flag & MINMAX_USEROT) {
 | 
						|
			/* take rotation of target into account by doing the transaction in target's localspace */
 | 
						|
			invert_m4_m4(imat, tarmat);
 | 
						|
			mul_m4_m4m4(tmat, obmat, imat);
 | 
						|
			copy_m4_m4(obmat, tmat);
 | 
						|
			unit_m4(tarmat);
 | 
						|
		}
 | 
						|
		
 | 
						|
		switch (data->minmaxflag) {
 | 
						|
		case TRACK_Z:
 | 
						|
			val1 = tarmat[3][2];
 | 
						|
			val2 = obmat[3][2]-data->offset;
 | 
						|
			index = 2;
 | 
						|
			break;
 | 
						|
		case TRACK_Y:
 | 
						|
			val1 = tarmat[3][1];
 | 
						|
			val2 = obmat[3][1]-data->offset;
 | 
						|
			index = 1;
 | 
						|
			break;
 | 
						|
		case TRACK_X:
 | 
						|
			val1 = tarmat[3][0];
 | 
						|
			val2 = obmat[3][0]-data->offset;
 | 
						|
			index = 0;
 | 
						|
			break;
 | 
						|
		case TRACK_nZ:
 | 
						|
			val2 = tarmat[3][2];
 | 
						|
			val1 = obmat[3][2]-data->offset;
 | 
						|
			index = 2;
 | 
						|
			break;
 | 
						|
		case TRACK_nY:
 | 
						|
			val2 = tarmat[3][1];
 | 
						|
			val1 = obmat[3][1]-data->offset;
 | 
						|
			index = 1;
 | 
						|
			break;
 | 
						|
		case TRACK_nX:
 | 
						|
			val2 = tarmat[3][0];
 | 
						|
			val1 = obmat[3][0]-data->offset;
 | 
						|
			index = 0;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		
 | 
						|
		if (val1 > val2) {
 | 
						|
			obmat[3][index] = tarmat[3][index] + data->offset;
 | 
						|
			if (data->flag & MINMAX_STICKY) {
 | 
						|
				if (data->flag & MINMAX_STUCK) {
 | 
						|
					VECCOPY(obmat[3], data->cache);
 | 
						|
				} 
 | 
						|
				else {
 | 
						|
					VECCOPY(data->cache, obmat[3]);
 | 
						|
					data->flag |= MINMAX_STUCK;
 | 
						|
				}
 | 
						|
			}
 | 
						|
			if (data->flag & MINMAX_USEROT) {
 | 
						|
				/* get out of localspace */
 | 
						|
				mul_m4_m4m4(tmat, obmat, ct->matrix);
 | 
						|
				copy_m4_m4(cob->matrix, tmat);
 | 
						|
			} 
 | 
						|
			else {			
 | 
						|
				VECCOPY(cob->matrix[3], obmat[3]);
 | 
						|
			}
 | 
						|
		} 
 | 
						|
		else {
 | 
						|
			data->flag &= ~MINMAX_STUCK;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_MINMAX = {
 | 
						|
	CONSTRAINT_TYPE_MINMAX, /* type */
 | 
						|
	sizeof(bMinMaxConstraint), /* size */
 | 
						|
	"Floor", /* name */
 | 
						|
	"bMinMaxConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	minmax_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	minmax_new_data, /* new data */
 | 
						|
	minmax_get_tars, /* get constraint targets */
 | 
						|
	minmax_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	minmax_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ------- RigidBody Joint ---------- */
 | 
						|
 | 
						|
static void rbj_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bRigidBodyJointConstraint *data= (bRigidBodyJointConstraint *)cdata;
 | 
						|
	
 | 
						|
	// removed code which set target of this constraint  
 | 
						|
	data->type=1;
 | 
						|
}
 | 
						|
 | 
						|
static void rbj_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bRigidBodyJointConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int rbj_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bRigidBodyJointConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints without subtargets */
 | 
						|
		SINGLETARGETNS_GET_TARS(con, data->tar, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void rbj_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bRigidBodyJointConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGETNS_FLUSH_TARS(con, data->tar, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_RIGIDBODYJOINT = {
 | 
						|
	CONSTRAINT_TYPE_RIGIDBODYJOINT, /* type */
 | 
						|
	sizeof(bRigidBodyJointConstraint), /* size */
 | 
						|
	"RigidBody Joint", /* name */
 | 
						|
	"bRigidBodyJointConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	rbj_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	rbj_new_data, /* new data */
 | 
						|
	rbj_get_tars, /* get constraint targets */
 | 
						|
	rbj_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	NULL /* evaluate - this is not solved here... is just an interface for game-engine */
 | 
						|
};
 | 
						|
 | 
						|
/* -------- Clamp To ---------- */
 | 
						|
 | 
						|
static void clampto_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bClampToConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int clampto_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bClampToConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints without subtargets */
 | 
						|
		SINGLETARGETNS_GET_TARS(con, data->tar, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void clampto_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bClampToConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGETNS_FLUSH_TARS(con, data->tar, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void clampto_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		Curve *cu= ct->tar->data;
 | 
						|
		
 | 
						|
		/* note: when creating constraints that follow path, the curve gets the CU_PATH set now,
 | 
						|
		 *		currently for paths to work it needs to go through the bevlist/displist system (ton) 
 | 
						|
		 */
 | 
						|
		
 | 
						|
		/* only happens on reload file, but violates depsgraph still... fix! */
 | 
						|
		if (cu->path==NULL || cu->path->data==NULL)
 | 
						|
			makeDispListCurveTypes(cob->scene, ct->tar, 0);
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* technically, this isn't really needed for evaluation, but we don't know what else
 | 
						|
	 * might end up calling this...
 | 
						|
	 */
 | 
						|
	if (ct)
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
}
 | 
						|
 | 
						|
static void clampto_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bClampToConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target and it is a curve */
 | 
						|
	if (VALID_CONS_TARGET(ct) && (ct->tar->type == OB_CURVE)) {
 | 
						|
		Curve *cu= data->tar->data;
 | 
						|
		float obmat[4][4], targetMatrix[4][4], ownLoc[3];
 | 
						|
		float curveMin[3], curveMax[3];
 | 
						|
		
 | 
						|
		copy_m4_m4(obmat, cob->matrix);
 | 
						|
		unit_m4(targetMatrix);
 | 
						|
		VECCOPY(ownLoc, obmat[3]);
 | 
						|
		
 | 
						|
		INIT_MINMAX(curveMin, curveMax)
 | 
						|
		minmax_object(ct->tar, curveMin, curveMax);
 | 
						|
		
 | 
						|
		/* get targetmatrix */
 | 
						|
		if (cu->path && cu->path->data) {
 | 
						|
			float vec[4], dir[3], totmat[4][4];
 | 
						|
			float curvetime;
 | 
						|
			short clamp_axis;
 | 
						|
			
 | 
						|
			/* find best position on curve */
 | 
						|
			/* 1. determine which axis to sample on? */
 | 
						|
			if (data->flag == CLAMPTO_AUTO) {
 | 
						|
				float size[3];
 | 
						|
				sub_v3_v3v3(size, curveMax, curveMin);
 | 
						|
				
 | 
						|
				/* find axis along which the bounding box has the greatest
 | 
						|
				 * extent. Otherwise, default to the x-axis, as that is quite
 | 
						|
				 * frequently used.
 | 
						|
				 */
 | 
						|
				if ((size[2]>size[0]) && (size[2]>size[1]))
 | 
						|
					clamp_axis= CLAMPTO_Z - 1;
 | 
						|
				else if ((size[1]>size[0]) && (size[1]>size[2]))
 | 
						|
					clamp_axis= CLAMPTO_Y - 1;
 | 
						|
				else
 | 
						|
					clamp_axis = CLAMPTO_X - 1;
 | 
						|
			}
 | 
						|
			else 
 | 
						|
				clamp_axis= data->flag - 1;
 | 
						|
				
 | 
						|
			/* 2. determine position relative to curve on a 0-1 scale based on bounding box */
 | 
						|
			if (data->flag2 & CLAMPTO_CYCLIC) {
 | 
						|
				/* cyclic, so offset within relative bounding box is used */
 | 
						|
				float len= (curveMax[clamp_axis] - curveMin[clamp_axis]);
 | 
						|
				float offset;
 | 
						|
				
 | 
						|
				/* check to make sure len is not so close to zero that it'll cause errors */
 | 
						|
				if (IS_EQ(len, 0) == 0) {
 | 
						|
					/* find bounding-box range where target is located */
 | 
						|
					if (ownLoc[clamp_axis] < curveMin[clamp_axis]) {
 | 
						|
						/* bounding-box range is before */
 | 
						|
						offset= curveMin[clamp_axis];
 | 
						|
						
 | 
						|
						while (ownLoc[clamp_axis] < offset)
 | 
						|
							offset -= len;
 | 
						|
						
 | 
						|
						/* now, we calculate as per normal, except using offset instead of curveMin[clamp_axis] */
 | 
						|
						curvetime = (ownLoc[clamp_axis] - offset) / (len);
 | 
						|
					}
 | 
						|
					else if (ownLoc[clamp_axis] > curveMax[clamp_axis]) {
 | 
						|
						/* bounding-box range is after */
 | 
						|
						offset= curveMax[clamp_axis];
 | 
						|
						
 | 
						|
						while (ownLoc[clamp_axis] > offset) {
 | 
						|
							if ((offset + len) > ownLoc[clamp_axis])
 | 
						|
								break;
 | 
						|
							else
 | 
						|
								offset += len;
 | 
						|
						}
 | 
						|
						
 | 
						|
						/* now, we calculate as per normal, except using offset instead of curveMax[clamp_axis] */
 | 
						|
						curvetime = (ownLoc[clamp_axis] - offset) / (len);
 | 
						|
					}
 | 
						|
					else {
 | 
						|
						/* as the location falls within bounds, just calculate */
 | 
						|
						curvetime = (ownLoc[clamp_axis] - curveMin[clamp_axis]) / (len);
 | 
						|
					}
 | 
						|
				}
 | 
						|
				else {
 | 
						|
					/* as length is close to zero, curvetime by default should be 0 (i.e. the start) */
 | 
						|
					curvetime= 0.0f;
 | 
						|
				}
 | 
						|
			}
 | 
						|
			else {
 | 
						|
				/* no cyclic, so position is clamped to within the bounding box */
 | 
						|
				if (ownLoc[clamp_axis] <= curveMin[clamp_axis])
 | 
						|
					curvetime = 0.0f;
 | 
						|
				else if (ownLoc[clamp_axis] >= curveMax[clamp_axis])
 | 
						|
					curvetime = 1.0f;
 | 
						|
				else if ( IS_EQ((curveMax[clamp_axis] - curveMin[clamp_axis]), 0) == 0 )
 | 
						|
					curvetime = (ownLoc[clamp_axis] - curveMin[clamp_axis]) / (curveMax[clamp_axis] - curveMin[clamp_axis]);
 | 
						|
				else 
 | 
						|
					curvetime = 0.0f;
 | 
						|
			}
 | 
						|
			
 | 
						|
			/* 3. position on curve */
 | 
						|
			if (where_on_path(ct->tar, curvetime, vec, dir, NULL, NULL) ) {
 | 
						|
				unit_m4(totmat);
 | 
						|
				VECCOPY(totmat[3], vec);
 | 
						|
				
 | 
						|
				mul_serie_m4(targetMatrix, ct->tar->obmat, totmat, NULL, NULL, NULL, NULL, NULL, NULL);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* obtain final object position */
 | 
						|
		VECCOPY(cob->matrix[3], targetMatrix[3]);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_CLAMPTO = {
 | 
						|
	CONSTRAINT_TYPE_CLAMPTO, /* type */
 | 
						|
	sizeof(bClampToConstraint), /* size */
 | 
						|
	"Clamp To", /* name */
 | 
						|
	"bClampToConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	clampto_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	clampto_get_tars, /* get constraint targets */
 | 
						|
	clampto_flush_tars, /* flush constraint targets */
 | 
						|
	clampto_get_tarmat, /* get target matrix */
 | 
						|
	clampto_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Transform Constraint ----------- */
 | 
						|
 | 
						|
static void transform_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bTransformConstraint *data= (bTransformConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->map[0]= 0;
 | 
						|
	data->map[1]= 1;
 | 
						|
	data->map[2]= 2;
 | 
						|
}
 | 
						|
 | 
						|
static void transform_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bTransformConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int transform_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bTransformConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void transform_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bTransformConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void transform_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bTransformConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float loc[3], eul[3], size[3];
 | 
						|
		float dvec[3], sval[3];
 | 
						|
		int i;
 | 
						|
		
 | 
						|
		/* obtain target effect */
 | 
						|
		switch (data->from) {
 | 
						|
			case 2: /* scale */
 | 
						|
				mat4_to_size( dvec,ct->matrix);
 | 
						|
				break;
 | 
						|
			case 1: /* rotation (convert to degrees first) */
 | 
						|
				mat4_to_eulO( dvec, cob->rotOrder,ct->matrix);
 | 
						|
				for (i=0; i<3; i++)
 | 
						|
					dvec[i] = (float)(dvec[i] / M_PI * 180);
 | 
						|
				break;
 | 
						|
			default: /* location */
 | 
						|
				copy_v3_v3(dvec, ct->matrix[3]);
 | 
						|
				break;
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* extract components of owner's matrix */
 | 
						|
		VECCOPY(loc, cob->matrix[3]);
 | 
						|
		mat4_to_eulO( eul, cob->rotOrder,cob->matrix);
 | 
						|
		mat4_to_size( size,cob->matrix);	
 | 
						|
		
 | 
						|
		/* determine where in range current transforms lie */
 | 
						|
		if (data->expo) {
 | 
						|
			for (i=0; i<3; i++) {
 | 
						|
				if (data->from_max[i] - data->from_min[i])
 | 
						|
					sval[i]= (dvec[i] - data->from_min[i]) / (data->from_max[i] - data->from_min[i]);
 | 
						|
				else
 | 
						|
					sval[i]= 0.0f;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else {
 | 
						|
			/* clamp transforms out of range */
 | 
						|
			for (i=0; i<3; i++) {
 | 
						|
				CLAMP(dvec[i], data->from_min[i], data->from_max[i]);
 | 
						|
				if (data->from_max[i] - data->from_min[i])
 | 
						|
					sval[i]= (dvec[i] - data->from_min[i]) / (data->from_max[i] - data->from_min[i]);
 | 
						|
				else
 | 
						|
					sval[i]= 0.0f;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		
 | 
						|
		
 | 
						|
		/* apply transforms */
 | 
						|
		switch (data->to) {
 | 
						|
			case 2: /* scaling */
 | 
						|
				for (i=0; i<3; i++)
 | 
						|
					size[i]= data->to_min[i] + (sval[(int)data->map[i]] * (data->to_max[i] - data->to_min[i])); 
 | 
						|
				break;
 | 
						|
			case 1: /* rotation */
 | 
						|
				for (i=0; i<3; i++) {
 | 
						|
					float tmin, tmax;
 | 
						|
					
 | 
						|
					tmin= data->to_min[i];
 | 
						|
					tmax= data->to_max[i];
 | 
						|
					
 | 
						|
					/* all values here should be in degrees */
 | 
						|
					eul[i]= tmin + (sval[(int)data->map[i]] * (tmax - tmin)); 
 | 
						|
					
 | 
						|
					/* now convert final value back to radians */
 | 
						|
					eul[i] = (float)(eul[i] / 180 * M_PI);
 | 
						|
				}
 | 
						|
				break;
 | 
						|
			default: /* location */
 | 
						|
				/* get new location */
 | 
						|
				for (i=0; i<3; i++)
 | 
						|
					loc[i]= (data->to_min[i] + (sval[(int)data->map[i]] * (data->to_max[i] - data->to_min[i])));
 | 
						|
				
 | 
						|
				/* add original location back on (so that it can still be moved) */
 | 
						|
				add_v3_v3v3(loc, cob->matrix[3], loc);
 | 
						|
				break;
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* apply to matrix */
 | 
						|
		loc_eulO_size_to_mat4(cob->matrix, loc, eul, size, cob->rotOrder);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_TRANSFORM = {
 | 
						|
	CONSTRAINT_TYPE_TRANSFORM, /* type */
 | 
						|
	sizeof(bTransformConstraint), /* size */
 | 
						|
	"Transform", /* name */
 | 
						|
	"bTransformConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	transform_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	transform_new_data, /* new data */
 | 
						|
	transform_get_tars, /* get constraint targets */
 | 
						|
	transform_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get a target matrix */
 | 
						|
	transform_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ---------- Shrinkwrap Constraint ----------- */
 | 
						|
 | 
						|
static void shrinkwrap_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bShrinkwrapConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->target, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int shrinkwrap_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bShrinkwrapConstraint *data = con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		SINGLETARGETNS_GET_TARS(con, data->target, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void shrinkwrap_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bShrinkwrapConstraint *data = con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		SINGLETARGETNS_FLUSH_TARS(con, data->target, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void shrinkwrap_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	bShrinkwrapConstraint *scon = (bShrinkwrapConstraint *) con->data;
 | 
						|
	
 | 
						|
	if( VALID_CONS_TARGET(ct) && (ct->tar->type == OB_MESH) )
 | 
						|
	{
 | 
						|
		int fail = FALSE;
 | 
						|
		float co[3] = {0.0f, 0.0f, 0.0f};
 | 
						|
		float no[3] = {0.0f, 0.0f, 0.0f};
 | 
						|
		float dist;
 | 
						|
 | 
						|
		SpaceTransform transform;
 | 
						|
		DerivedMesh *target = object_get_derived_final(cob->scene, ct->tar, CD_MASK_BAREMESH);
 | 
						|
		BVHTreeRayHit hit;
 | 
						|
		BVHTreeNearest nearest;
 | 
						|
 | 
						|
		BVHTreeFromMesh treeData;
 | 
						|
		memset( &treeData, 0, sizeof(treeData) );
 | 
						|
 | 
						|
		nearest.index = -1;
 | 
						|
		nearest.dist = FLT_MAX;
 | 
						|
 | 
						|
		hit.index = -1;
 | 
						|
		hit.dist = 100000.0f;  //TODO should use FLT_MAX.. but normal projection doenst yet supports it
 | 
						|
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
 | 
						|
		if(target != NULL)
 | 
						|
		{
 | 
						|
			space_transform_from_matrixs(&transform, cob->matrix, ct->tar->obmat);
 | 
						|
 | 
						|
			switch(scon->shrinkType)
 | 
						|
			{
 | 
						|
				case MOD_SHRINKWRAP_NEAREST_SURFACE:
 | 
						|
				case MOD_SHRINKWRAP_NEAREST_VERTEX:
 | 
						|
 | 
						|
					if(scon->shrinkType == MOD_SHRINKWRAP_NEAREST_VERTEX)
 | 
						|
						bvhtree_from_mesh_verts(&treeData, target, 0.0, 2, 6);
 | 
						|
					else
 | 
						|
						bvhtree_from_mesh_faces(&treeData, target, 0.0, 2, 6);
 | 
						|
 | 
						|
					if(treeData.tree == NULL)
 | 
						|
					{
 | 
						|
						fail = TRUE;
 | 
						|
						break;
 | 
						|
					}
 | 
						|
 | 
						|
					space_transform_apply(&transform, co);
 | 
						|
 | 
						|
					BLI_bvhtree_find_nearest(treeData.tree, co, &nearest, treeData.nearest_callback, &treeData);
 | 
						|
					
 | 
						|
					dist = len_v3v3(co, nearest.co);
 | 
						|
					interp_v3_v3v3(co, co, nearest.co, (dist - scon->dist)/dist);	/* linear interpolation */
 | 
						|
					space_transform_invert(&transform, co);
 | 
						|
				break;
 | 
						|
 | 
						|
				case MOD_SHRINKWRAP_PROJECT:
 | 
						|
					if(scon->projAxis & MOD_SHRINKWRAP_PROJECT_OVER_X_AXIS) no[0] = 1.0f;
 | 
						|
					if(scon->projAxis & MOD_SHRINKWRAP_PROJECT_OVER_Y_AXIS) no[1] = 1.0f;
 | 
						|
					if(scon->projAxis & MOD_SHRINKWRAP_PROJECT_OVER_Z_AXIS) no[2] = 1.0f;
 | 
						|
 | 
						|
					if(INPR(no,no) < FLT_EPSILON)
 | 
						|
					{
 | 
						|
						fail = TRUE;
 | 
						|
						break;
 | 
						|
					}
 | 
						|
 | 
						|
					normalize_v3(no);
 | 
						|
 | 
						|
 | 
						|
					bvhtree_from_mesh_faces(&treeData, target, scon->dist, 4, 6);
 | 
						|
					if(treeData.tree == NULL)
 | 
						|
					{
 | 
						|
						fail = TRUE;
 | 
						|
						break;
 | 
						|
					}
 | 
						|
 | 
						|
					if(normal_projection_project_vertex(0, co, no, &transform, treeData.tree, &hit, treeData.raycast_callback, &treeData) == FALSE)
 | 
						|
					{
 | 
						|
						fail = TRUE;
 | 
						|
						break;
 | 
						|
					}
 | 
						|
					VECCOPY(co, hit.co);
 | 
						|
				break;
 | 
						|
			}
 | 
						|
 | 
						|
			free_bvhtree_from_mesh(&treeData);
 | 
						|
 | 
						|
			target->release(target);
 | 
						|
 | 
						|
			if(fail == TRUE)
 | 
						|
			{
 | 
						|
				/* Don't move the point */
 | 
						|
				co[0] = co[1] = co[2] = 0.0f;
 | 
						|
			}
 | 
						|
 | 
						|
			/* co is in local object coordinates, change it to global and update target position */
 | 
						|
			mul_v3_m4v3(co, cob->matrix, co);
 | 
						|
			VECCOPY(ct->matrix[3], co);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void shrinkwrap_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	/* only evaluate if there is a target */
 | 
						|
	if (VALID_CONS_TARGET(ct))
 | 
						|
	{
 | 
						|
		VECCOPY(cob->matrix[3], ct->matrix[3]);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_SHRINKWRAP = {
 | 
						|
	CONSTRAINT_TYPE_SHRINKWRAP, /* type */
 | 
						|
	sizeof(bShrinkwrapConstraint), /* size */
 | 
						|
	"Shrinkwrap", /* name */
 | 
						|
	"bShrinkwrapConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	shrinkwrap_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	shrinkwrap_get_tars, /* get constraint targets */
 | 
						|
	shrinkwrap_flush_tars, /* flush constraint targets */
 | 
						|
	shrinkwrap_get_tarmat, /* get a target matrix */
 | 
						|
	shrinkwrap_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* --------- Damped Track ---------- */
 | 
						|
 | 
						|
static void damptrack_new_data (void *cdata)
 | 
						|
{
 | 
						|
	bDampTrackConstraint *data= (bDampTrackConstraint *)cdata;
 | 
						|
	
 | 
						|
	data->trackflag = TRACK_Y;
 | 
						|
}	
 | 
						|
 | 
						|
static void damptrack_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bDampTrackConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int damptrack_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bDampTrackConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_GET_TARS(con, data->tar, data->subtarget, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void damptrack_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bDampTrackConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGET_FLUSH_TARS(con, data->tar, data->subtarget, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* array of direction vectors for the tracking flags */
 | 
						|
static const float track_dir_vecs[6][3] = {
 | 
						|
	{+1,0,0}, {0,+1,0}, {0,0,+1},		/* TRACK_X,  TRACK_Y,  TRACK_Z */
 | 
						|
	{-1,0,0}, {0,-1,0}, {0,0,-1}		/* TRACK_NX, TRACK_NY, TRACK_NZ */
 | 
						|
};
 | 
						|
 | 
						|
static void damptrack_evaluate (bConstraint *con, bConstraintOb *cob, ListBase *targets)
 | 
						|
{
 | 
						|
	bDampTrackConstraint *data= con->data;
 | 
						|
	bConstraintTarget *ct= targets->first;
 | 
						|
	
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		float obvec[3], tarvec[3], obloc[3];
 | 
						|
		float raxis[3], rangle;
 | 
						|
		float rmat[3][3], tmat[4][4];
 | 
						|
		
 | 
						|
		/* find the (unit) direction that the axis we're interested in currently points 
 | 
						|
		 *	- mul_mat3_m4_v3() only takes the 3x3 (rotation+scaling) components of the 4x4 matrix 
 | 
						|
		 *	- the normalisation step at the end should take care of any unwanted scaling
 | 
						|
		 *	  left over in the 3x3 matrix we used
 | 
						|
		 */
 | 
						|
		VECCOPY(obvec, track_dir_vecs[data->trackflag]);
 | 
						|
		mul_mat3_m4_v3(cob->matrix, obvec);
 | 
						|
		
 | 
						|
		if (normalize_v3(obvec) == 0.0f) {
 | 
						|
			/* exceptional case - just use the track vector as appropriate */
 | 
						|
			VECCOPY(obvec, track_dir_vecs[data->trackflag]);
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* find the (unit) direction vector going from the owner to the target */
 | 
						|
		VECCOPY(obloc, cob->matrix[3]);
 | 
						|
		sub_v3_v3v3(tarvec, ct->matrix[3], obloc);
 | 
						|
		
 | 
						|
		if (normalize_v3(tarvec) == 0.0f) {
 | 
						|
			/* the target is sitting on the owner, so just make them use the same direction vectors */
 | 
						|
			// FIXME: or would it be better to use the pure direction vector?
 | 
						|
			VECCOPY(tarvec, obvec);
 | 
						|
			//VECCOPY(tarvec, track_dir_vecs[data->trackflag]);
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* determine the axis-angle rotation, which represents the smallest possible rotation
 | 
						|
		 * between the two rotation vectors (i.e. the 'damping' referred to in the name)
 | 
						|
		 *	- we take this to be the rotation around the normal axis/vector to the plane defined
 | 
						|
		 *	  by the current and destination vectors, which will 'map' the current axis to the 
 | 
						|
		 *	  destination vector
 | 
						|
		 *	- the min/max wrappers around (obvec . tarvec) result (stored temporarily in rangle)
 | 
						|
		 *	  are used to ensure that the smallest angle is chosen
 | 
						|
		 */
 | 
						|
		cross_v3_v3v3(raxis, obvec, tarvec);
 | 
						|
		
 | 
						|
		rangle= dot_v3v3(obvec, tarvec);
 | 
						|
		rangle= acos( MAX2(-1.0f, MIN2(1.0f, rangle)) );
 | 
						|
		
 | 
						|
		/* construct rotation matrix from the axis-angle rotation found above 
 | 
						|
		 *	- this call takes care to make sure that the axis provided is a unit vector first
 | 
						|
		 */
 | 
						|
		axis_angle_to_mat3( rmat,raxis, rangle);
 | 
						|
		
 | 
						|
		/* rotate the owner in the way defined by this rotation matrix, then reapply the location since
 | 
						|
		 * we may have destroyed that in the process of multiplying the matrix
 | 
						|
		 */
 | 
						|
		unit_m4(tmat);
 | 
						|
		mul_m4_m3m4(tmat, rmat, cob->matrix); // m1, m3, m2
 | 
						|
		
 | 
						|
		copy_m4_m4(cob->matrix, tmat);
 | 
						|
		VECCOPY(cob->matrix[3], obloc);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_DAMPTRACK = {
 | 
						|
	CONSTRAINT_TYPE_DAMPTRACK, /* type */
 | 
						|
	sizeof(bDampTrackConstraint), /* size */
 | 
						|
	"Damped Track", /* name */
 | 
						|
	"bDampTrackConstraint", /* struct name */
 | 
						|
	NULL, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	damptrack_id_looper, /* id looper */
 | 
						|
	NULL, /* copy data */
 | 
						|
	damptrack_new_data, /* new data */
 | 
						|
	damptrack_get_tars, /* get constraint targets */
 | 
						|
	damptrack_flush_tars, /* flush constraint targets */
 | 
						|
	default_get_tarmat, /* get target matrix */
 | 
						|
	damptrack_evaluate /* evaluate */
 | 
						|
};
 | 
						|
 | 
						|
/* ----------- Spline IK ------------ */
 | 
						|
 | 
						|
static void splineik_free (bConstraint *con)
 | 
						|
{
 | 
						|
	bSplineIKConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* binding array */
 | 
						|
	if (data->points)
 | 
						|
		MEM_freeN(data->points);
 | 
						|
}	
 | 
						|
 | 
						|
static void splineik_copy (bConstraint *con, bConstraint *srccon)
 | 
						|
{
 | 
						|
	bSplineIKConstraint *src= srccon->data;
 | 
						|
	bSplineIKConstraint *dst= con->data;
 | 
						|
	
 | 
						|
	/* copy the binding array */
 | 
						|
	dst->points= MEM_dupallocN(src->points);
 | 
						|
}
 | 
						|
 | 
						|
static void splineik_id_looper (bConstraint *con, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bSplineIKConstraint *data= con->data;
 | 
						|
	
 | 
						|
	/* target only */
 | 
						|
	func(con, (ID**)&data->tar, userdata);
 | 
						|
}
 | 
						|
 | 
						|
static int splineik_get_tars (bConstraint *con, ListBase *list)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bSplineIKConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct;
 | 
						|
		
 | 
						|
		/* standard target-getting macro for single-target constraints without subtargets */
 | 
						|
		SINGLETARGETNS_GET_TARS(con, data->tar, ct, list)
 | 
						|
		
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void splineik_flush_tars (bConstraint *con, ListBase *list, short nocopy)
 | 
						|
{
 | 
						|
	if (con && list) {
 | 
						|
		bSplineIKConstraint *data= con->data;
 | 
						|
		bConstraintTarget *ct= list->first;
 | 
						|
		
 | 
						|
		/* the following macro is used for all standard single-target constraints */
 | 
						|
		SINGLETARGETNS_FLUSH_TARS(con, data->tar, ct, list, nocopy)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void splineik_get_tarmat (bConstraint *con, bConstraintOb *cob, bConstraintTarget *ct, float ctime)
 | 
						|
{
 | 
						|
	if (VALID_CONS_TARGET(ct)) {
 | 
						|
		Curve *cu= ct->tar->data;
 | 
						|
		
 | 
						|
		/* note: when creating constraints that follow path, the curve gets the CU_PATH set now,
 | 
						|
		 *		currently for paths to work it needs to go through the bevlist/displist system (ton) 
 | 
						|
		 */
 | 
						|
		
 | 
						|
		/* only happens on reload file, but violates depsgraph still... fix! */
 | 
						|
		if (cu->path==NULL || cu->path->data==NULL)
 | 
						|
			makeDispListCurveTypes(cob->scene, ct->tar, 0);
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* technically, this isn't really needed for evaluation, but we don't know what else
 | 
						|
	 * might end up calling this...
 | 
						|
	 */
 | 
						|
	if (ct)
 | 
						|
		unit_m4(ct->matrix);
 | 
						|
}
 | 
						|
 | 
						|
static bConstraintTypeInfo CTI_SPLINEIK = {
 | 
						|
	CONSTRAINT_TYPE_SPLINEIK, /* type */
 | 
						|
	sizeof(bSplineIKConstraint), /* size */
 | 
						|
	"Spline IK", /* name */
 | 
						|
	"bSplineIKConstraint", /* struct name */
 | 
						|
	splineik_free, /* free data */
 | 
						|
	NULL, /* relink data */
 | 
						|
	splineik_id_looper, /* id looper */
 | 
						|
	splineik_copy, /* copy data */
 | 
						|
	NULL, /* new data */
 | 
						|
	splineik_get_tars, /* get constraint targets */
 | 
						|
	splineik_flush_tars, /* flush constraint targets */
 | 
						|
	splineik_get_tarmat, /* get target matrix */
 | 
						|
	NULL /* evaluate - solved as separate loop */
 | 
						|
};
 | 
						|
 | 
						|
/* ************************* Constraints Type-Info *************************** */
 | 
						|
/* All of the constraints api functions use bConstraintTypeInfo structs to carry out
 | 
						|
 * and operations that involve constraint specific code.
 | 
						|
 */
 | 
						|
 | 
						|
/* These globals only ever get directly accessed in this file */
 | 
						|
static bConstraintTypeInfo *constraintsTypeInfo[NUM_CONSTRAINT_TYPES];
 | 
						|
static short CTI_INIT= 1; /* when non-zero, the list needs to be updated */
 | 
						|
 | 
						|
/* This function only gets called when CTI_INIT is non-zero */
 | 
						|
static void constraints_init_typeinfo () {
 | 
						|
	constraintsTypeInfo[0]=  NULL; 					/* 'Null' Constraint */
 | 
						|
	constraintsTypeInfo[1]=  &CTI_CHILDOF; 			/* ChildOf Constraint */
 | 
						|
	constraintsTypeInfo[2]=  &CTI_TRACKTO;			/* TrackTo Constraint */
 | 
						|
	constraintsTypeInfo[3]=  &CTI_KINEMATIC;		/* IK Constraint */
 | 
						|
	constraintsTypeInfo[4]=  &CTI_FOLLOWPATH;		/* Follow-Path Constraint */
 | 
						|
	constraintsTypeInfo[5]=  &CTI_ROTLIMIT;			/* Limit Rotation Constraint */
 | 
						|
	constraintsTypeInfo[6]=  &CTI_LOCLIMIT;			/* Limit Location Constraint */
 | 
						|
	constraintsTypeInfo[7]=  &CTI_SIZELIMIT;		/* Limit Scaling Constraint */
 | 
						|
	constraintsTypeInfo[8]=  &CTI_ROTLIKE;			/* Copy Rotation Constraint */
 | 
						|
	constraintsTypeInfo[9]=  &CTI_LOCLIKE;			/* Copy Location Constraint */
 | 
						|
	constraintsTypeInfo[10]= &CTI_SIZELIKE;			/* Copy Scaling Constraint */
 | 
						|
	constraintsTypeInfo[11]= &CTI_PYTHON;			/* Python/Script Constraint */
 | 
						|
	constraintsTypeInfo[12]= &CTI_ACTION;			/* Action Constraint */
 | 
						|
	constraintsTypeInfo[13]= &CTI_LOCKTRACK;		/* Locked-Track Constraint */
 | 
						|
	constraintsTypeInfo[14]= &CTI_DISTLIMIT;		/* Limit Distance Constraint */
 | 
						|
	constraintsTypeInfo[15]= &CTI_STRETCHTO; 		/* StretchTo Constaint */ 
 | 
						|
	constraintsTypeInfo[16]= &CTI_MINMAX;  			/* Floor Constraint */
 | 
						|
	constraintsTypeInfo[17]= &CTI_RIGIDBODYJOINT;	/* RigidBody Constraint */
 | 
						|
	constraintsTypeInfo[18]= &CTI_CLAMPTO; 			/* ClampTo Constraint */	
 | 
						|
	constraintsTypeInfo[19]= &CTI_TRANSFORM;		/* Transformation Constraint */
 | 
						|
	constraintsTypeInfo[20]= &CTI_SHRINKWRAP;		/* Shrinkwrap Constraint */
 | 
						|
	constraintsTypeInfo[21]= &CTI_DAMPTRACK;		/* Damped TrackTo Constraint */
 | 
						|
	constraintsTypeInfo[22]= &CTI_SPLINEIK;			/* Spline IK Constraint */
 | 
						|
	constraintsTypeInfo[23]= &CTI_TRANSLIKE;		/* Copy Transforms Constraint */
 | 
						|
	constraintsTypeInfo[24]= &CTI_SAMEVOL;			/* Maintain Volume Constraint */
 | 
						|
}
 | 
						|
 | 
						|
/* This function should be used for getting the appropriate type-info when only
 | 
						|
 * a constraint type is known
 | 
						|
 */
 | 
						|
bConstraintTypeInfo *get_constraint_typeinfo (int type)
 | 
						|
{
 | 
						|
	/* initialise the type-info list? */
 | 
						|
	if (CTI_INIT) {
 | 
						|
		constraints_init_typeinfo();
 | 
						|
		CTI_INIT = 0;
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* only return for valid types */
 | 
						|
	if ( (type >= CONSTRAINT_TYPE_NULL) && 
 | 
						|
		 (type < NUM_CONSTRAINT_TYPES ) ) 
 | 
						|
	{
 | 
						|
		/* there shouldn't be any segfaults here... */
 | 
						|
		return constraintsTypeInfo[type];
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		printf("No valid constraint type-info data available. Type = %i \n", type);
 | 
						|
	}
 | 
						|
	
 | 
						|
	return NULL;
 | 
						|
} 
 | 
						|
 
 | 
						|
/* This function should always be used to get the appropriate type-info, as it
 | 
						|
 * has checks which prevent segfaults in some weird cases.
 | 
						|
 */
 | 
						|
bConstraintTypeInfo *constraint_get_typeinfo (bConstraint *con)
 | 
						|
{
 | 
						|
	/* only return typeinfo for valid constraints */
 | 
						|
	if (con)
 | 
						|
		return get_constraint_typeinfo(con->type);
 | 
						|
	else
 | 
						|
		return NULL;
 | 
						|
}
 | 
						|
 | 
						|
/* ************************* General Constraints API ************************** */
 | 
						|
/* The functions here are called by various parts of Blender. Very few (should be none if possible)
 | 
						|
 * constraint-specific code should occur here.
 | 
						|
 */
 | 
						|
 
 | 
						|
/* ---------- Data Management ------- */
 | 
						|
 | 
						|
/* Free data of a specific constraint if it has any info */
 | 
						|
void free_constraint_data (bConstraint *con)
 | 
						|
{
 | 
						|
	if (con->data) {
 | 
						|
		bConstraintTypeInfo *cti= constraint_get_typeinfo(con);
 | 
						|
		
 | 
						|
		/* perform any special freeing constraint may have */
 | 
						|
		if (cti && cti->free_data)
 | 
						|
			cti->free_data(con);
 | 
						|
		
 | 
						|
		/* free constraint data now */
 | 
						|
		MEM_freeN(con->data);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Free all constraints from a constraint-stack */
 | 
						|
void free_constraints (ListBase *list)
 | 
						|
{
 | 
						|
	bConstraint *con;
 | 
						|
	
 | 
						|
	/* Free constraint data and also any extra data */
 | 
						|
	for (con= list->first; con; con= con->next)
 | 
						|
		free_constraint_data(con);
 | 
						|
	
 | 
						|
	/* Free the whole list */
 | 
						|
	BLI_freelistN(list);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/* Remove the specified constraint from the given constraint stack */
 | 
						|
int remove_constraint (ListBase *list, bConstraint *con)
 | 
						|
{
 | 
						|
	if (con) {
 | 
						|
		free_constraint_data(con);
 | 
						|
		BLI_freelinkN(list, con);
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	else
 | 
						|
		return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Remove the nth constraint from the given constraint stack */
 | 
						|
int remove_constraint_index (ListBase *list, int index)
 | 
						|
{
 | 
						|
	bConstraint *con= BLI_findlink(list, index);
 | 
						|
	
 | 
						|
	if (con)
 | 
						|
		return remove_constraint(list, con);
 | 
						|
	else 
 | 
						|
		return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Remove all the constraints of the specified type from the given constraint stack */
 | 
						|
void remove_constraints_type (ListBase *list, short type, short last_only)
 | 
						|
{
 | 
						|
	bConstraint *con, *conp;
 | 
						|
	
 | 
						|
	if (list == NULL)
 | 
						|
		return;
 | 
						|
	
 | 
						|
	/* remove from the end of the list to make it faster to find the last instance */
 | 
						|
	for (con= list->last; con; con= conp) {
 | 
						|
		conp= con->prev;
 | 
						|
		
 | 
						|
		if (con->type == type) {
 | 
						|
			remove_constraint(list, con);
 | 
						|
			if (last_only) 
 | 
						|
				return;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* ......... */
 | 
						|
 | 
						|
/* Creates a new constraint, initialises its data, and returns it */
 | 
						|
static bConstraint *add_new_constraint_internal (const char *name, short type)
 | 
						|
{
 | 
						|
	bConstraint *con= MEM_callocN(sizeof(bConstraint), "Constraint");
 | 
						|
	bConstraintTypeInfo *cti= get_constraint_typeinfo(type);
 | 
						|
	const char *newName;
 | 
						|
 | 
						|
	/* Set up a generic constraint datablock */
 | 
						|
	con->type = type;
 | 
						|
	con->flag |= CONSTRAINT_EXPAND;
 | 
						|
	con->enforce = 1.0f;
 | 
						|
 | 
						|
	/* Determine a basic name, and info */
 | 
						|
	if (cti) {
 | 
						|
		/* initialise constraint data */
 | 
						|
		con->data = MEM_callocN(cti->size, cti->structName);
 | 
						|
		
 | 
						|
		/* only constraints that change any settings need this */
 | 
						|
		if (cti->new_data)
 | 
						|
			cti->new_data(con->data);
 | 
						|
		
 | 
						|
		/* if no name is provided, use the type of the constraint as the name */
 | 
						|
		newName= (name && name[0]) ? name : cti->name;
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		/* if no name is provided, use the generic "Const" name */
 | 
						|
		// NOTE: any constraint type that gets here really shouldn't get added...
 | 
						|
		newName= (name && name[0]) ? name : "Const";
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* copy the name */
 | 
						|
	BLI_strncpy(con->name, newName, sizeof(con->name));
 | 
						|
	
 | 
						|
	/* return the new constraint */
 | 
						|
	return con;
 | 
						|
}
 | 
						|
 | 
						|
/* if pchan is not NULL then assume we're adding a pose constraint */
 | 
						|
static bConstraint *add_new_constraint (Object *ob, bPoseChannel *pchan, const char *name, short type)
 | 
						|
{
 | 
						|
	bConstraint *con;
 | 
						|
	ListBase *list;
 | 
						|
	
 | 
						|
	/* add the constraint */
 | 
						|
	con= add_new_constraint_internal(name, type);
 | 
						|
	
 | 
						|
	/* find the constraint stack - bone or object? */
 | 
						|
	list = (pchan) ? (&pchan->constraints) : (&ob->constraints);
 | 
						|
	
 | 
						|
	if (list) {
 | 
						|
		/* add new constraint to end of list of constraints before ensuring that it has a unique name
 | 
						|
		 * (otherwise unique-naming code will fail, since it assumes element exists in list)
 | 
						|
		 */
 | 
						|
		BLI_addtail(list, con);
 | 
						|
		unique_constraint_name(con, list);
 | 
						|
		
 | 
						|
		/* if the target list is a list on some PoseChannel belonging to a proxy-protected
 | 
						|
		 * Armature layer, we must tag newly added constraints with a flag which allows them
 | 
						|
		 * to persist after proxy syncing has been done
 | 
						|
		 */
 | 
						|
		if (proxylocked_constraints_owner(ob, pchan))
 | 
						|
			con->flag |= CONSTRAINT_PROXY_LOCAL;
 | 
						|
		
 | 
						|
		/* make this constraint the active one */
 | 
						|
		constraints_set_active(list, con);
 | 
						|
	}
 | 
						|
	
 | 
						|
	return con;
 | 
						|
}
 | 
						|
 | 
						|
/* ......... */
 | 
						|
 | 
						|
/* Add new constraint for the given bone */
 | 
						|
bConstraint *add_pose_constraint (Object *ob, bPoseChannel *pchan, const char *name, short type)
 | 
						|
{
 | 
						|
	if (pchan == NULL)
 | 
						|
		return NULL;
 | 
						|
	
 | 
						|
	return add_new_constraint(ob, pchan, name, type);
 | 
						|
}
 | 
						|
 | 
						|
/* Add new constraint for the given object */
 | 
						|
bConstraint *add_ob_constraint(Object *ob, const char *name, short type)
 | 
						|
{
 | 
						|
	return add_new_constraint(ob, NULL, name, type);
 | 
						|
}
 | 
						|
 | 
						|
/* ......... */
 | 
						|
 | 
						|
/* Reassign links that constraints have to other data (called during file loading?) */
 | 
						|
void relink_constraints (ListBase *conlist)
 | 
						|
{
 | 
						|
	bConstraint *con;
 | 
						|
	bConstraintTarget *ct;
 | 
						|
	
 | 
						|
	for (con= conlist->first; con; con= con->next) {
 | 
						|
		bConstraintTypeInfo *cti= constraint_get_typeinfo(con);
 | 
						|
		
 | 
						|
		if (cti) {
 | 
						|
			/* relink any targets */
 | 
						|
			if (cti->get_constraint_targets) {
 | 
						|
				ListBase targets = {NULL, NULL};
 | 
						|
				
 | 
						|
				cti->get_constraint_targets(con, &targets);
 | 
						|
				for (ct= targets.first; ct; ct= ct->next) {
 | 
						|
					ID_NEW(ct->tar);
 | 
						|
				}
 | 
						|
				
 | 
						|
				if (cti->flush_constraint_targets)
 | 
						|
					cti->flush_constraint_targets(con, &targets, 0);
 | 
						|
			}
 | 
						|
			
 | 
						|
			/* relink any other special data */
 | 
						|
			if (cti->relink_data)
 | 
						|
				cti->relink_data(con);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Run the given callback on all ID-blocks in list of constraints */
 | 
						|
void id_loop_constraints (ListBase *conlist, ConstraintIDFunc func, void *userdata)
 | 
						|
{
 | 
						|
	bConstraint *con;
 | 
						|
	
 | 
						|
	for (con= conlist->first; con; con= con->next) {
 | 
						|
		bConstraintTypeInfo *cti= constraint_get_typeinfo(con);
 | 
						|
		
 | 
						|
		if (cti) {
 | 
						|
			if (cti->id_looper)
 | 
						|
				cti->id_looper(con, func, userdata);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* ......... */
 | 
						|
 | 
						|
/* helper for copy_constraints(), to be used for making sure that ID's are valid */
 | 
						|
static void con_extern_cb(bConstraint *con, ID **idpoin, void *userdata)
 | 
						|
{
 | 
						|
	if (*idpoin && (*idpoin)->lib)
 | 
						|
		id_lib_extern(*idpoin);
 | 
						|
}
 | 
						|
 | 
						|
/* duplicate all of the constraints in a constraint stack */
 | 
						|
void copy_constraints (ListBase *dst, const ListBase *src)
 | 
						|
{
 | 
						|
	bConstraint *con, *srccon;
 | 
						|
	
 | 
						|
	dst->first= dst->last= NULL;
 | 
						|
	BLI_duplicatelist(dst, src);
 | 
						|
	
 | 
						|
	for (con=dst->first, srccon=src->first; con && srccon; srccon=srccon->next, con=con->next) {
 | 
						|
		bConstraintTypeInfo *cti= constraint_get_typeinfo(con);
 | 
						|
		
 | 
						|
		/* make a new copy of the constraint's data */
 | 
						|
		con->data = MEM_dupallocN(con->data);
 | 
						|
		
 | 
						|
		/* only do specific constraints if required */
 | 
						|
		if (cti) {
 | 
						|
			/* perform custom copying operations if needed */
 | 
						|
			if (cti->copy_data)
 | 
						|
				cti->copy_data(con, srccon);
 | 
						|
			
 | 
						|
			/* go over used ID-links for this constraint to ensure that they are valid for proxies */
 | 
						|
			if (cti->id_looper)
 | 
						|
				cti->id_looper(con, con_extern_cb, NULL);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* ......... */
 | 
						|
 | 
						|
/* finds the 'active' constraint in a constraint stack */
 | 
						|
bConstraint *constraints_get_active (ListBase *list)
 | 
						|
{
 | 
						|
	bConstraint *con;
 | 
						|
	
 | 
						|
	/* search for the first constraint with the 'active' flag set */
 | 
						|
	if (list) {
 | 
						|
		for (con= list->first; con; con= con->next) {
 | 
						|
			if (con->flag & CONSTRAINT_ACTIVE)
 | 
						|
				return con;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	
 | 
						|
	/* no active constraint found */
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
/* Set the given constraint as the active one (clearing all the others) */
 | 
						|
void constraints_set_active (ListBase *list, bConstraint *con)
 | 
						|
{
 | 
						|
	bConstraint *c;
 | 
						|
	
 | 
						|
	if (list) {
 | 
						|
		for (c= list->first; c; c= c->next) {
 | 
						|
			if (c == con) 
 | 
						|
				c->flag |= CONSTRAINT_ACTIVE;
 | 
						|
			else 
 | 
						|
				c->flag &= ~CONSTRAINT_ACTIVE;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* -------- Constraints and Proxies ------- */
 | 
						|
 | 
						|
/* Rescue all constraints tagged as being CONSTRAINT_PROXY_LOCAL (i.e. added to bone that's proxy-synced in this file) */
 | 
						|
void extract_proxylocal_constraints (ListBase *dst, ListBase *src)
 | 
						|
{
 | 
						|
	bConstraint *con, *next;
 | 
						|
	
 | 
						|
	/* for each tagged constraint, remove from src and move to dst */
 | 
						|
	for (con= src->first; con; con= next) {
 | 
						|
		next= con->next;
 | 
						|
		
 | 
						|
		/* check if tagged */
 | 
						|
		if (con->flag & CONSTRAINT_PROXY_LOCAL) {
 | 
						|
			BLI_remlink(src, con);
 | 
						|
			BLI_addtail(dst, con);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Returns if the owner of the constraint is proxy-protected */
 | 
						|
short proxylocked_constraints_owner (Object *ob, bPoseChannel *pchan)
 | 
						|
{
 | 
						|
	/* Currently, constraints can only be on object or bone level */
 | 
						|
	if (ob && ob->proxy) {
 | 
						|
		if (ob->pose && pchan) {
 | 
						|
			bArmature *arm= ob->data;
 | 
						|
			
 | 
						|
			/* On bone-level, check if bone is on proxy-protected layer */
 | 
						|
			if ((pchan->bone) && (pchan->bone->layer & arm->layer_protected))
 | 
						|
				return 1;
 | 
						|
		}
 | 
						|
		else {
 | 
						|
			/* FIXME: constraints on object-level are not handled well yet */
 | 
						|
			return 1;
 | 
						|
		}	
 | 
						|
	}
 | 
						|
	
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* -------- Target-Matrix Stuff ------- */
 | 
						|
 | 
						|
/* This function is a relic from the prior implementations of the constraints system, when all
 | 
						|
 * constraints either had one or no targets. It used to be called during the main constraint solving
 | 
						|
 * loop, but is now only used for the remaining cases for a few constraints. 
 | 
						|
 *
 | 
						|
 * None of the actual calculations of the matricies should be done here! Also, this function is 
 | 
						|
 * not to be used by any new constraints, particularly any that have multiple targets.
 | 
						|
 */
 | 
						|
void get_constraint_target_matrix (struct Scene *scene, bConstraint *con, int n, short ownertype, void *ownerdata, float mat[][4], float ctime)
 | 
						|
{
 | 
						|
	bConstraintTypeInfo *cti= constraint_get_typeinfo(con);
 | 
						|
	ListBase targets = {NULL, NULL};
 | 
						|
	bConstraintOb *cob;
 | 
						|
	bConstraintTarget *ct;
 | 
						|
	
 | 
						|
	if (cti && cti->get_constraint_targets) {
 | 
						|
		/* make 'constraint-ob' */
 | 
						|
		cob= MEM_callocN(sizeof(bConstraintOb), "tempConstraintOb");
 | 
						|
		cob->type= ownertype;
 | 
						|
		cob->scene = scene;
 | 
						|
		switch (ownertype) {
 | 
						|
			case CONSTRAINT_OBTYPE_OBJECT: /* it is usually this case */
 | 
						|
			{
 | 
						|
				cob->ob= (Object *)ownerdata;
 | 
						|
				cob->pchan= NULL;
 | 
						|
				if (cob->ob) {
 | 
						|
					copy_m4_m4(cob->matrix, cob->ob->obmat);
 | 
						|
					copy_m4_m4(cob->startmat, cob->matrix);
 | 
						|
				}
 | 
						|
				else {
 | 
						|
					unit_m4(cob->matrix);
 | 
						|
					unit_m4(cob->startmat);
 | 
						|
				}
 | 
						|
			}	
 | 
						|
				break;
 | 
						|
			case CONSTRAINT_OBTYPE_BONE: /* this may occur in some cases */
 | 
						|
			{
 | 
						|
				cob->ob= NULL; /* this might not work at all :/ */
 | 
						|
				cob->pchan= (bPoseChannel *)ownerdata;
 | 
						|
				if (cob->pchan) {
 | 
						|
					copy_m4_m4(cob->matrix, cob->pchan->pose_mat);
 | 
						|
					copy_m4_m4(cob->startmat, cob->matrix);
 | 
						|
				}
 | 
						|
				else {
 | 
						|
					unit_m4(cob->matrix);
 | 
						|
					unit_m4(cob->startmat);
 | 
						|
				}
 | 
						|
			}
 | 
						|
				break;
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* get targets - we only need the first one though (and there should only be one) */
 | 
						|
		cti->get_constraint_targets(con, &targets);
 | 
						|
		
 | 
						|
		/* only calculate the target matrix on the first target */
 | 
						|
		ct= (bConstraintTarget *)targets.first;
 | 
						|
		while(ct && n-- > 0)
 | 
						|
			ct= ct->next;
 | 
						|
 | 
						|
		if (ct) {
 | 
						|
			if (cti->get_target_matrix)
 | 
						|
				cti->get_target_matrix(con, cob, ct, ctime);
 | 
						|
			copy_m4_m4(mat, ct->matrix);
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* free targets + 'constraint-ob' */
 | 
						|
		if (cti->flush_constraint_targets)
 | 
						|
			cti->flush_constraint_targets(con, &targets, 1);
 | 
						|
		MEM_freeN(cob);
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		/* invalid constraint - perhaps... */
 | 
						|
		unit_m4(mat);
 | 
						|
	}
 | 
						|
}
 | 
						|
 
 | 
						|
/* ---------- Evaluation ----------- */
 | 
						|
 | 
						|
/* This function is called whenever constraints need to be evaluated. Currently, all
 | 
						|
 * constraints that can be evaluated are everytime this gets run.
 | 
						|
 *
 | 
						|
 * constraints_make_evalob and constraints_clear_evalob should be called before and 
 | 
						|
 * after running this function, to sort out cob
 | 
						|
 */
 | 
						|
void solve_constraints (ListBase *conlist, bConstraintOb *cob, float ctime)
 | 
						|
{
 | 
						|
	bConstraint *con;
 | 
						|
	float solution[4][4], delta[4][4];
 | 
						|
	float oldmat[4][4], imat[4][4];
 | 
						|
	float enf;
 | 
						|
 | 
						|
	/* check that there is a valid constraint object to evaluate */
 | 
						|
	if (cob == NULL)
 | 
						|
		return;
 | 
						|
	
 | 
						|
	/* loop over available constraints, solving and blending them */
 | 
						|
	for (con= conlist->first; con; con= con->next) {
 | 
						|
		bConstraintTypeInfo *cti= constraint_get_typeinfo(con);
 | 
						|
		ListBase targets = {NULL, NULL};
 | 
						|
		
 | 
						|
		/* these we can skip completely (invalid constraints...) */
 | 
						|
		if (cti == NULL) continue;
 | 
						|
		if (con->flag & (CONSTRAINT_DISABLE|CONSTRAINT_OFF)) continue;
 | 
						|
		/* these constraints can't be evaluated anyway */
 | 
						|
		if (cti->evaluate_constraint == NULL) continue;
 | 
						|
		/* influence == 0 should be ignored */
 | 
						|
		if (con->enforce == 0.0f) continue;
 | 
						|
		
 | 
						|
		/* influence of constraint
 | 
						|
		 * 	- value should have been set from animation data already
 | 
						|
		 */
 | 
						|
		enf = con->enforce;
 | 
						|
		
 | 
						|
		/* move owner matrix into right space */
 | 
						|
		constraint_mat_convertspace(cob->ob, cob->pchan, cob->matrix, CONSTRAINT_SPACE_WORLD, con->ownspace);
 | 
						|
		copy_m4_m4(oldmat, cob->matrix);
 | 
						|
		
 | 
						|
		/* prepare targets for constraint solving */
 | 
						|
		if (cti->get_constraint_targets) {
 | 
						|
			bConstraintTarget *ct;
 | 
						|
			
 | 
						|
			/* get targets 
 | 
						|
			 * 	- constraints should use ct->matrix, not directly accessing values
 | 
						|
			 *	- ct->matrix members have not yet been calculated here! 
 | 
						|
			 */
 | 
						|
			cti->get_constraint_targets(con, &targets);
 | 
						|
			
 | 
						|
			/* set matrices 
 | 
						|
			 * 	- calculate if possible, otherwise just initialise as identity matrix 
 | 
						|
			 */
 | 
						|
			if (cti->get_target_matrix) {
 | 
						|
				for (ct= targets.first; ct; ct= ct->next) 
 | 
						|
					cti->get_target_matrix(con, cob, ct, ctime);
 | 
						|
			}
 | 
						|
			else {
 | 
						|
				for (ct= targets.first; ct; ct= ct->next)
 | 
						|
					unit_m4(ct->matrix);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* Solve the constraint */
 | 
						|
		cti->evaluate_constraint(con, cob, &targets);
 | 
						|
		
 | 
						|
		/* clear targets after use 
 | 
						|
		 *	- this should free temp targets but no data should be copied back
 | 
						|
		 *	  as constraints may have done some nasty things to it...
 | 
						|
		 */
 | 
						|
		if (cti->flush_constraint_targets) {
 | 
						|
			cti->flush_constraint_targets(con, &targets, 1);
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* Interpolate the enforcement, to blend result of constraint into final owner transform */
 | 
						|
		/* 1. Remove effects of original matrix from constraint solution ==> delta */
 | 
						|
		invert_m4_m4(imat, oldmat);
 | 
						|
		copy_m4_m4(solution, cob->matrix);
 | 
						|
		mul_m4_m4m4(delta, solution, imat);
 | 
						|
		
 | 
						|
		/* 2. If constraint influence is not full strength, then interpolate
 | 
						|
		 * 	identity_matrix --> delta_matrix to get the effect the constraint actually exerts
 | 
						|
		 */
 | 
						|
		if (enf < 1.0) {
 | 
						|
			float identity[4][4];
 | 
						|
			unit_m4(identity);
 | 
						|
			blend_m4_m4m4(delta, identity, delta, enf);
 | 
						|
		}
 | 
						|
		
 | 
						|
		/* 3. Now multiply the delta by the matrix in use before the evaluation */
 | 
						|
		mul_m4_m4m4(cob->matrix, delta, oldmat);
 | 
						|
		
 | 
						|
		/* move owner back into worldspace for next constraint/other business */
 | 
						|
		if ((con->flag & CONSTRAINT_SPACEONCE) == 0) 
 | 
						|
			constraint_mat_convertspace(cob->ob, cob->pchan, cob->matrix, con->ownspace, CONSTRAINT_SPACE_WORLD);
 | 
						|
	}
 | 
						|
}
 |