| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2010-03-21 01:14:04 +00:00
										 |  |  |  * $Id$ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ***** 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, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2004 by Blender Foundation | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): none yet. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | 	meshtools.c: no editmode (violated already :), tools operating on meshes | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2009-09-06 13:20:05 +00:00
										 |  |  | #include <float.h>
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | #include "DNA_key_types.h"
 | 
					
						
							|  |  |  | #include "DNA_material_types.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							|  |  |  | #include "DNA_object_types.h"
 | 
					
						
							|  |  |  | #include "DNA_scene_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "BLI_blenlib.h"
 | 
					
						
							|  |  |  | #include "BLI_editVert.h"
 | 
					
						
							|  |  |  | #include "BLI_ghash.h"
 | 
					
						
							|  |  |  | #include "BLI_rand.h" /* for randome face sorting */
 | 
					
						
							|  |  |  | #include "BLI_threads.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | #include "BKE_context.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "BKE_depsgraph.h"
 | 
					
						
							| 
									
										
										
										
											2010-08-22 14:15:28 +00:00
										 |  |  | #include "BKE_deform.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | #include "BKE_DerivedMesh.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | #include "BKE_key.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "BKE_library.h"
 | 
					
						
							|  |  |  | #include "BKE_main.h"
 | 
					
						
							|  |  |  | #include "BKE_mesh.h"
 | 
					
						
							|  |  |  | #include "BKE_material.h"
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | #include "BKE_report.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-25 08:03:05 +00:00
										 |  |  | #include "BKE_multires.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLO_sys_types.h" // for intptr_t support
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "ED_mesh.h"
 | 
					
						
							|  |  |  | #include "ED_object.h"
 | 
					
						
							| 
									
										
										
										
											2009-01-05 15:19:31 +00:00
										 |  |  | #include "ED_view3d.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | #include "WM_api.h"
 | 
					
						
							|  |  |  | #include "WM_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* own include */ | 
					
						
							| 
									
										
										
										
											2009-01-01 13:15:35 +00:00
										 |  |  | #include "mesh_intern.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* * ********************** no editmode!!! *********** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | /*********************** JOIN ***************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* join selected meshes into the active mesh, context sensitive
 | 
					
						
							|  |  |  | return 0 if no join is made (error) and 1 of the join is done */ | 
					
						
							| 
									
										
										
										
											2009-07-11 10:20:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 01:36:14 +00:00
										 |  |  | int join_mesh_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-01 12:47:49 +00:00
										 |  |  | 	Main *bmain= CTX_data_main(C); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	Scene *scene= CTX_data_scene(C); | 
					
						
							|  |  |  | 	Object *ob= CTX_data_active_object(C); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Material **matar, *ma; | 
					
						
							|  |  |  | 	Mesh *me; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	MVert *mvert, *mv, *mvertmain; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MEdge *medge = NULL, *medgemain; | 
					
						
							|  |  |  | 	MFace *mface = NULL, *mfacemain; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	Key *key, *nkey=NULL; | 
					
						
							|  |  |  | 	KeyBlock *kb, *okb, *kbn; | 
					
						
							|  |  |  | 	float imat[4][4], cmat[4][4], *fp1, *fp2, curpos; | 
					
						
							|  |  |  | 	int a, b, totcol, totmat=0, totedge=0, totvert=0, totface=0, ok=0; | 
					
						
							| 
									
										
										
										
											2009-11-10 06:29:10 +00:00
										 |  |  | 	int vertofs, *matmap=NULL; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	int	i, j, index, haskey=0, edgeofs, faceofs; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	bDeformGroup *dg, *odg; | 
					
						
							|  |  |  | 	MDeformVert *dvert; | 
					
						
							|  |  |  | 	CustomData vdata, edata, fdata; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	if(scene->obedit) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	/* ob is the object we are adding geometry to */ | 
					
						
							|  |  |  | 	if(!ob || ob->type!=OB_MESH) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* count & check */ | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	CTX_DATA_BEGIN(C, Base*, base, selected_editable_bases) { | 
					
						
							|  |  |  | 		if(base->object->type==OB_MESH) { | 
					
						
							|  |  |  | 			me= base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			totvert+= me->totvert; | 
					
						
							|  |  |  | 			totedge+= me->totedge; | 
					
						
							|  |  |  | 			totface+= me->totface; | 
					
						
							|  |  |  | 			totmat+= base->object->totcol; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if(base->object == ob) | 
					
						
							|  |  |  | 				ok= 1; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			/* check for shapekeys */ | 
					
						
							|  |  |  | 			if(me->key) | 
					
						
							|  |  |  | 				haskey++; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	CTX_DATA_END; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* that way the active object is always selected */  | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	if(ok==0) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	/* only join meshes if there are verts to join, there aren't too many, and we only had one mesh selected */ | 
					
						
							|  |  |  | 	me= (Mesh *)ob->data; | 
					
						
							|  |  |  | 	key= me->key; | 
					
						
							|  |  |  | 	if(totvert==0 || totvert>MESH_MAX_VERTS || totvert==me->totvert)  | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* new material indices and material array */ | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	matar= MEM_callocN(sizeof(void*)*totmat, "join_mesh matar"); | 
					
						
							| 
									
										
										
										
											2009-11-10 06:29:10 +00:00
										 |  |  | 	if (totmat) matmap= MEM_callocN(sizeof(int)*totmat, "join_mesh matmap"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	totcol= ob->totcol; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* obact materials in new main array, is nicer start! */ | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	for(a=0; a<ob->totcol; a++) { | 
					
						
							|  |  |  | 		matar[a]= give_current_material(ob, a+1); | 
					
						
							|  |  |  | 		id_us_plus((ID *)matar[a]); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/* increase id->us : will be lowered later */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	/* - if destination mesh had shapekeys, move them somewhere safe, and set up placeholders
 | 
					
						
							|  |  |  | 	 * 	with arrays that are large enough to hold shapekey data for all meshes | 
					
						
							|  |  |  | 	 * -	if destination mesh didn't have shapekeys, but we encountered some in the meshes we're  | 
					
						
							|  |  |  | 	 *	joining, set up a new keyblock and assign to the mesh | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if(key) { | 
					
						
							|  |  |  | 		/* make a duplicate copy that will only be used here... (must remember to free it!) */ | 
					
						
							|  |  |  | 		nkey= copy_key(key); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* for all keys in old block, clear data-arrays */ | 
					
						
							|  |  |  | 		for(kb= key->block.first; kb; kb= kb->next) { | 
					
						
							|  |  |  | 			if(kb->data) MEM_freeN(kb->data); | 
					
						
							|  |  |  | 			kb->data= MEM_callocN(sizeof(float)*3*totvert, "join_shapekey"); | 
					
						
							|  |  |  | 			kb->totelem= totvert; | 
					
						
							|  |  |  | 			kb->weights= NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(haskey) { | 
					
						
							|  |  |  | 		/* add a new key-block and add to the mesh */ | 
					
						
							|  |  |  | 		key= me->key= add_key((ID *)me); | 
					
						
							|  |  |  | 		key->type = KEY_RELATIVE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* first pass over objects - copying materials and vertexgroups across */ | 
					
						
							|  |  |  | 	CTX_DATA_BEGIN(C, Base*, base, selected_editable_bases) { | 
					
						
							|  |  |  | 		/* only act if a mesh, and not the one we're joining to */ | 
					
						
							|  |  |  | 		if((ob!=base->object) && (base->object->type==OB_MESH)) { | 
					
						
							|  |  |  | 			me= base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			/* Join this object's vertex groups to the base one's */ | 
					
						
							|  |  |  | 			for(dg=base->object->defbase.first; dg; dg=dg->next) { | 
					
						
							|  |  |  | 				/* See if this group exists in the object (if it doesn't, add it to the end) */ | 
					
						
							| 
									
										
										
										
											2010-08-22 14:15:28 +00:00
										 |  |  | 				if(!defgroup_find_name(ob, dg->name)) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 					odg = MEM_callocN(sizeof(bDeformGroup), "join deformGroup"); | 
					
						
							|  |  |  | 					memcpy(odg, dg, sizeof(bDeformGroup)); | 
					
						
							|  |  |  | 					BLI_addtail(&ob->defbase, odg); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(ob->defbase.first && ob->actdef==0) | 
					
						
							|  |  |  | 				ob->actdef=1; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if(me->totvert) { | 
					
						
							|  |  |  | 				/* Add this object's materials to the base one's if they don't exist already (but only if limits not exceeded yet) */ | 
					
						
							|  |  |  | 				if(totcol < MAXMAT-1) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					for(a=1; a<=base->object->totcol; a++) { | 
					
						
							|  |  |  | 						ma= give_current_material(base->object, a); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						for(b=0; b<totcol; b++) { | 
					
						
							|  |  |  | 							if(ma == matar[b]) break; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						if(b==totcol) { | 
					
						
							|  |  |  | 							matar[b]= ma; | 
					
						
							|  |  |  | 							if(ma) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								ma->id.us++; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							totcol++; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						if(totcol>=MAXMAT-1)  | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				/* if this mesh has shapekeys, check if destination mesh already has matching entries too */ | 
					
						
							|  |  |  | 				if(me->key && key) { | 
					
						
							|  |  |  | 					for(kb= me->key->block.first; kb; kb= kb->next) { | 
					
						
							|  |  |  | 						/* if key doesn't exist in destination mesh, add it */ | 
					
						
							|  |  |  | 						if(key_get_named_keyblock(key, kb->name) == NULL) { | 
					
						
							|  |  |  | 							/* copy this existing one over to the new shapekey block */ | 
					
						
							|  |  |  | 							kbn= MEM_dupallocN(kb); | 
					
						
							|  |  |  | 							kbn->prev= kbn->next= NULL; | 
					
						
							|  |  |  | 							 | 
					
						
							|  |  |  | 							/* adjust adrcode and other settings to fit (allocate a new data-array) */ | 
					
						
							|  |  |  | 							kbn->data= MEM_callocN(sizeof(float)*3*totvert, "joined_shapekey"); | 
					
						
							|  |  |  | 							kbn->totelem= totvert; | 
					
						
							|  |  |  | 							kbn->weights= NULL; | 
					
						
							|  |  |  | 							 | 
					
						
							|  |  |  | 							okb= key->block.last; | 
					
						
							|  |  |  | 							curpos= (okb) ? okb->pos : -0.1f; | 
					
						
							|  |  |  | 							if(key->type == KEY_RELATIVE) | 
					
						
							|  |  |  | 								kbn->pos= curpos + 0.1f; | 
					
						
							|  |  |  | 							else | 
					
						
							|  |  |  | 								kbn->pos= curpos; | 
					
						
							|  |  |  | 							 | 
					
						
							|  |  |  | 							BLI_addtail(&key->block, kbn); | 
					
						
							|  |  |  | 							kbn->adrcode= key->totkey; | 
					
						
							|  |  |  | 							key->totkey++; | 
					
						
							|  |  |  | 							if(key->totkey==1) key->refkey= kbn; | 
					
						
							|  |  |  | 							 | 
					
						
							|  |  |  | 							// XXX 2.5 Animato
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 							/* also, copy corresponding ipo-curve to ipo-block if applicable */ | 
					
						
							|  |  |  | 							if(me->key->ipo && key->ipo) { | 
					
						
							|  |  |  | 								// FIXME... this is a luxury item!
 | 
					
						
							|  |  |  | 								puts("FIXME: ignoring IPO's when joining shapekeys on Meshes for now..."); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	CTX_DATA_END; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* setup new data for destination mesh */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	memset(&vdata, 0, sizeof(vdata)); | 
					
						
							|  |  |  | 	memset(&edata, 0, sizeof(edata)); | 
					
						
							|  |  |  | 	memset(&fdata, 0, sizeof(fdata)); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	mvert= CustomData_add_layer(&vdata, CD_MVERT, CD_CALLOC, NULL, totvert); | 
					
						
							|  |  |  | 	medge= CustomData_add_layer(&edata, CD_MEDGE, CD_CALLOC, NULL, totedge); | 
					
						
							|  |  |  | 	mface= CustomData_add_layer(&fdata, CD_MFACE, CD_CALLOC, NULL, totface); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	mvertmain= mvert; | 
					
						
							|  |  |  | 	medgemain= medge; | 
					
						
							|  |  |  | 	mfacemain= mface; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	vertofs= 0; | 
					
						
							|  |  |  | 	edgeofs= 0; | 
					
						
							|  |  |  | 	faceofs= 0; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* inverse transform for all selected meshes in this object */ | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 	invert_m4_m4(imat, ob->obmat); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	CTX_DATA_BEGIN(C, Base*, base, selected_editable_bases) { | 
					
						
							|  |  |  | 		/* only join if this is a mesh */ | 
					
						
							|  |  |  | 		if(base->object->type==OB_MESH) { | 
					
						
							|  |  |  | 			me= base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if(me->totvert) { | 
					
						
							|  |  |  | 				/* standard data */ | 
					
						
							|  |  |  | 				CustomData_merge(&me->vdata, &vdata, CD_MASK_MESH, CD_DEFAULT, totvert); | 
					
						
							|  |  |  | 				CustomData_copy_data(&me->vdata, &vdata, 0, vertofs, me->totvert); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* vertex groups */ | 
					
						
							|  |  |  | 				dvert= CustomData_get(&vdata, vertofs, CD_MDEFORMVERT); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* NB: vertex groups here are new version */ | 
					
						
							|  |  |  | 				if(dvert) { | 
					
						
							|  |  |  | 					for(i=0; i<me->totvert; i++) { | 
					
						
							|  |  |  | 						for(j=0; j<dvert[i].totweight; j++) { | 
					
						
							|  |  |  | 							/*	Find the old vertex group */ | 
					
						
							|  |  |  | 							odg = BLI_findlink(&base->object->defbase, dvert[i].dw[j].def_nr); | 
					
						
							|  |  |  | 							if(odg) { | 
					
						
							|  |  |  | 								/*	Search for a match in the new object, and set new index */ | 
					
						
							|  |  |  | 								for(dg=ob->defbase.first, index=0; dg; dg=dg->next, index++) { | 
					
						
							|  |  |  | 									if(!strcmp(dg->name, odg->name)) { | 
					
						
							|  |  |  | 										dvert[i].dw[j].def_nr = index; | 
					
						
							|  |  |  | 										break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 									} | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* if this is the object we're merging into, no need to do anything */ | 
					
						
							|  |  |  | 				if(base->object != ob) { | 
					
						
							|  |  |  | 					/* watch this: switch matmul order really goes wrong */ | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 					mul_m4_m4m4(cmat, base->object->obmat, imat); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 					 | 
					
						
							|  |  |  | 					/* transform vertex coordinates into new space */ | 
					
						
							|  |  |  | 					for(a=0, mv=mvert; a < me->totvert; a++, mv++) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 						mul_m4_v3(cmat, mv->co); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					 | 
					
						
							|  |  |  | 					/* for each shapekey in destination mesh:
 | 
					
						
							|  |  |  | 					 *	- if there's a matching one, copy it across (will need to transform vertices into new space...) | 
					
						
							|  |  |  | 					 *	- otherwise, just copy own coordinates of mesh (no need to transform vertex coordinates into new space) | 
					
						
							|  |  |  | 					 */ | 
					
						
							|  |  |  | 					if(key) { | 
					
						
							|  |  |  | 						/* if this mesh has any shapekeys, check first, otherwise just copy coordinates */ | 
					
						
							|  |  |  | 						for(kb= key->block.first; kb; kb= kb->next) { | 
					
						
							|  |  |  | 							/* get pointer to where to write data for this mesh in shapekey's data array */ | 
					
						
							|  |  |  | 							fp1= ((float *)kb->data) + (vertofs*3);	 | 
					
						
							|  |  |  | 							 | 
					
						
							|  |  |  | 							/* check if this mesh has such a shapekey */ | 
					
						
							|  |  |  | 							okb= key_get_named_keyblock(me->key, kb->name); | 
					
						
							|  |  |  | 							if(okb) { | 
					
						
							|  |  |  | 								/* copy this mesh's shapekey to the destination shapekey (need to transform first) */ | 
					
						
							|  |  |  | 								fp2= ((float *)(okb->data)); | 
					
						
							|  |  |  | 								for(a=0; a < me->totvert; a++, fp1+=3, fp2+=3) { | 
					
						
							|  |  |  | 									VECCOPY(fp1, fp2); | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 									mul_m4_v3(cmat, fp1); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								/* copy this mesh's vertex coordinates to the destination shapekey */ | 
					
						
							|  |  |  | 								mv= mvert; | 
					
						
							|  |  |  | 								for(a=0; a < me->totvert; a++, fp1+=3, mv++) { | 
					
						
							|  |  |  | 									VECCOPY(fp1, mv->co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					/* for each shapekey in destination mesh:
 | 
					
						
							|  |  |  | 					 *	- if it was an 'original', copy the appropriate data from nkey | 
					
						
							|  |  |  | 					 *	- otherwise, copy across plain coordinates (no need to transform coordinates) | 
					
						
							|  |  |  | 					 */ | 
					
						
							|  |  |  | 					if(key) { | 
					
						
							|  |  |  | 						for(kb= key->block.first; kb; kb= kb->next) { | 
					
						
							|  |  |  | 							/* get pointer to where to write data for this mesh in shapekey's data array */ | 
					
						
							|  |  |  | 							fp1= ((float *)kb->data) + (vertofs*3);	 | 
					
						
							|  |  |  | 							 | 
					
						
							|  |  |  | 							/* check if this was one of the original shapekeys */ | 
					
						
							|  |  |  | 							okb= key_get_named_keyblock(nkey, kb->name); | 
					
						
							|  |  |  | 							if(okb) { | 
					
						
							|  |  |  | 								/* copy this mesh's shapekey to the destination shapekey */ | 
					
						
							|  |  |  | 								fp2= ((float *)(okb->data)); | 
					
						
							|  |  |  | 								for(a=0; a < me->totvert; a++, fp1+=3, fp2+=3) { | 
					
						
							|  |  |  | 									VECCOPY(fp1, fp2); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								/* copy base-coordinates to the destination shapekey */ | 
					
						
							|  |  |  | 								mv= mvert; | 
					
						
							|  |  |  | 								for(a=0; a < me->totvert; a++, fp1+=3, mv++) { | 
					
						
							|  |  |  | 									VECCOPY(fp1, mv->co); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* advance mvert pointer to end of base mesh's data */ | 
					
						
							|  |  |  | 				mvert+= me->totvert; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if(me->totface) { | 
					
						
							|  |  |  | 				/* make mapping for materials */ | 
					
						
							|  |  |  | 				for(a=1; a<=base->object->totcol; a++) { | 
					
						
							|  |  |  | 					ma= give_current_material(base->object, a); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					for(b=0; b<totcol; b++) { | 
					
						
							|  |  |  | 						if(ma == matar[b]) { | 
					
						
							|  |  |  | 							matmap[a-1]= b; | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2010-10-25 08:03:05 +00:00
										 |  |  | 				if(base->object!=ob) | 
					
						
							|  |  |  | 					multiresModifier_prepare_join(scene, base->object, ob); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				CustomData_merge(&me->fdata, &fdata, CD_MASK_MESH, CD_DEFAULT, totface); | 
					
						
							|  |  |  | 				CustomData_copy_data(&me->fdata, &fdata, 0, faceofs, me->totface); | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				for(a=0; a<me->totface; a++, mface++) { | 
					
						
							|  |  |  | 					mface->v1+= vertofs; | 
					
						
							|  |  |  | 					mface->v2+= vertofs; | 
					
						
							|  |  |  | 					mface->v3+= vertofs; | 
					
						
							|  |  |  | 					if(mface->v4) mface->v4+= vertofs; | 
					
						
							|  |  |  | 					 | 
					
						
							| 
									
										
										
										
											2009-11-10 06:29:10 +00:00
										 |  |  | 					if (matmap) | 
					
						
							|  |  |  | 						mface->mat_nr= matmap[(int)mface->mat_nr]; | 
					
						
							|  |  |  | 					else  | 
					
						
							|  |  |  | 						mface->mat_nr= 0; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				faceofs += me->totface; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if(me->totedge) { | 
					
						
							|  |  |  | 				CustomData_merge(&me->edata, &edata, CD_MASK_MESH, CD_DEFAULT, totedge); | 
					
						
							|  |  |  | 				CustomData_copy_data(&me->edata, &edata, 0, edgeofs, me->totedge); | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				for(a=0; a<me->totedge; a++, medge++) { | 
					
						
							|  |  |  | 					medge->v1+= vertofs; | 
					
						
							|  |  |  | 					medge->v2+= vertofs; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				edgeofs += me->totedge; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			/* vertofs is used to help newly added verts be reattached to their edge/face 
 | 
					
						
							|  |  |  | 			 * (cannot be set earlier, or else reattaching goes wrong) | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			vertofs += me->totvert; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			/* free base, now that data is merged */ | 
					
						
							|  |  |  | 			if(base->object != ob) | 
					
						
							| 
									
										
										
										
											2010-08-01 12:47:49 +00:00
										 |  |  | 				ED_base_object_free_and_unlink(bmain, scene, base); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	CTX_DATA_END; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	/* return to mesh we're merging to */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	me= ob->data; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	CustomData_free(&me->vdata, me->totvert); | 
					
						
							|  |  |  | 	CustomData_free(&me->edata, me->totedge); | 
					
						
							|  |  |  | 	CustomData_free(&me->fdata, me->totface); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me->totvert= totvert; | 
					
						
							|  |  |  | 	me->totedge= totedge; | 
					
						
							|  |  |  | 	me->totface= totface; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	me->vdata= vdata; | 
					
						
							|  |  |  | 	me->edata= edata; | 
					
						
							|  |  |  | 	me->fdata= fdata; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mesh_update_customdata_pointers(me); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* old material array */ | 
					
						
							|  |  |  | 	for(a=1; a<=ob->totcol; a++) { | 
					
						
							|  |  |  | 		ma= ob->mat[a-1]; | 
					
						
							|  |  |  | 		if(ma) ma->id.us--; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for(a=1; a<=me->totcol; a++) { | 
					
						
							|  |  |  | 		ma= me->mat[a-1]; | 
					
						
							|  |  |  | 		if(ma) ma->id.us--; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(ob->mat) MEM_freeN(ob->mat); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	if(ob->matbits) MEM_freeN(ob->matbits); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(me->mat) MEM_freeN(me->mat); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	ob->mat= me->mat= NULL; | 
					
						
							|  |  |  | 	ob->matbits= NULL; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if(totcol) { | 
					
						
							|  |  |  | 		me->mat= matar; | 
					
						
							|  |  |  | 		ob->mat= MEM_callocN(sizeof(void *)*totcol, "join obmatar"); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		ob->matbits= MEM_callocN(sizeof(char)*totcol, "join obmatbits"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		MEM_freeN(matar); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	ob->totcol= me->totcol= totcol; | 
					
						
							|  |  |  | 	ob->colbits= 0; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 06:29:10 +00:00
										 |  |  | 	if (matmap) MEM_freeN(matmap); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* other mesh users */ | 
					
						
							|  |  |  | 	test_object_materials((ID *)me); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	/* free temp copy of destination shapekeys (if applicable) */ | 
					
						
							|  |  |  | 	if(nkey) { | 
					
						
							|  |  |  | 		// XXX 2.5 Animato
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 		/* free it's ipo too - both are not actually freed from memory yet as ID-blocks */ | 
					
						
							|  |  |  | 		if(nkey->ipo) { | 
					
						
							|  |  |  | 			free_ipo(nkey->ipo); | 
					
						
							| 
									
										
										
										
											2010-08-01 12:47:49 +00:00
										 |  |  | 			BLI_remlink(&bmain->ipo, nkey->ipo); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			MEM_freeN(nkey->ipo); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		free_key(nkey); | 
					
						
							| 
									
										
										
										
											2010-08-01 12:47:49 +00:00
										 |  |  | 		BLI_remlink(&bmain->key, nkey); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		MEM_freeN(nkey); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2010-08-01 12:47:49 +00:00
										 |  |  | 	DAG_scene_sort(bmain, scene);	// removed objects, need to rebuild dag before editmode call
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	ED_object_enter_editmode(C, EM_WAITCURSOR); | 
					
						
							| 
									
										
										
											
												- add torus back from 2.4x as an operator
  bpy.ops.mesh.primitive_torus_add(major_radius=1, minor_radius=0.25, major_segments=48, minor_segments=16)
- experemental dynamic menus, used for INFO_MT_file, INFO_MT_file_import, INFO_MT_file_export and INFO_MT_mesh_add. these can have items added from python.
eg.
- removed OBJECT_OT_mesh_add, use the python add menu instead.
- made mesh primitive ops -  MESH_OT_primitive_plane_add, ...cube_add, etc. work in object mode.
- RNA scene.active_object wrapped
- bugfix [#19466] 2.5: Tweak menu only available for mesh objects added within Edit Mode
  ED_object_exit_editmode was always doing an undo push, made this optional using the existing flag - EM_DO_UNDO, called everywhere except when adding primitives.
											
										 
											2009-10-10 21:23:20 +00:00
										 |  |  | 	ED_object_exit_editmode(C, EM_FREEDATA|EM_WAITCURSOR|EM_DO_UNDO); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_SCENE|ND_OB_ACTIVE, scene); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | /*********************** JOIN AS SHAPES ***************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Append selected meshes vertex locations as shapes of the active mesh, 
 | 
					
						
							|  |  |  |   return 0 if no join is made (error) and 1 of the join is done */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int join_mesh_shapes_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Scene *scene= CTX_data_scene(C); | 
					
						
							|  |  |  | 	Object *ob= CTX_data_active_object(C); | 
					
						
							|  |  |  | 	Mesh *me= (Mesh *)ob->data; | 
					
						
							|  |  |  | 	Mesh *selme=NULL; | 
					
						
							|  |  |  | 	DerivedMesh *dm=NULL; | 
					
						
							|  |  |  | 	Key *key=me->key; | 
					
						
							|  |  |  | 	KeyBlock *kb; | 
					
						
							|  |  |  | 	int ok=0, nonequal_verts=0; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	CTX_DATA_BEGIN(C, Base*, base, selected_editable_bases) { | 
					
						
							|  |  |  | 		if (base->object == ob) continue; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (base->object->type==OB_MESH) { | 
					
						
							|  |  |  | 			selme = (Mesh *)base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if (selme->totvert==me->totvert) | 
					
						
							|  |  |  | 				ok++; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				nonequal_verts=1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	CTX_DATA_END; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (!ok) { | 
					
						
							|  |  |  | 		if (nonequal_verts) | 
					
						
							|  |  |  | 			BKE_report(op->reports, RPT_ERROR, "Selected meshes must have equal numbers of vertices."); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			BKE_report(op->reports, RPT_ERROR, "No additional selected meshes with equal vertex count to join."); | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(key == NULL) { | 
					
						
							|  |  |  | 		key= me->key= add_key((ID *)me); | 
					
						
							|  |  |  | 		key->type= KEY_RELATIVE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* first key added, so it was the basis. initialise it with the existing mesh */ | 
					
						
							| 
									
										
										
										
											2009-12-28 18:03:04 +00:00
										 |  |  | 		kb= add_keyblock(key, NULL); | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 		mesh_to_key(me, kb); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* now ready to add new keys from selected meshes */ | 
					
						
							|  |  |  | 	CTX_DATA_BEGIN(C, Base*, base, selected_editable_bases) { | 
					
						
							|  |  |  | 		if (base->object == ob) continue; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if(base->object->type==OB_MESH) { | 
					
						
							|  |  |  | 			selme = (Mesh *)base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			if (selme->totvert==me->totvert) { | 
					
						
							|  |  |  | 				dm = mesh_get_derived_deform(scene, base->object, CD_MASK_BAREMESH); | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (!dm) continue; | 
					
						
							|  |  |  | 					 | 
					
						
							| 
									
										
										
										
											2009-12-28 18:03:04 +00:00
										 |  |  | 				kb= add_keyblock(key, base->object->id.name+2); | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 				 | 
					
						
							|  |  |  | 				DM_to_meshkey(dm, me, kb); | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				dm->release(dm); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	CTX_DATA_END; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_SCENE|ND_OB_ACTIVE, scene); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* ********************* MESH VERTEX OCTREE LOOKUP ************* */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* important note; this is unfinished, needs better API for editmode, and custom threshold */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MOC_RES			8
 | 
					
						
							|  |  |  | #define MOC_NODE_RES	8
 | 
					
						
							| 
									
										
										
										
											2010-01-04 14:59:53 +00:00
										 |  |  | #define MOC_THRESH		0.00002f
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct MocNode { | 
					
						
							|  |  |  | 	struct MocNode *next; | 
					
						
							|  |  |  | 	intptr_t index[MOC_NODE_RES]; | 
					
						
							|  |  |  | } MocNode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int mesh_octree_get_base_offs(float *co, float *offs, float *div) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int vx, vy, vz; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	vx= floor( (co[0]-offs[0])/div[0] ); | 
					
						
							|  |  |  | 	vy= floor( (co[1]-offs[1])/div[1] ); | 
					
						
							|  |  |  | 	vz= floor( (co[2]-offs[2])/div[2] ); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	CLAMP(vx, 0, MOC_RES-1); | 
					
						
							|  |  |  | 	CLAMP(vy, 0, MOC_RES-1); | 
					
						
							|  |  |  | 	CLAMP(vz, 0, MOC_RES-1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (vx*MOC_RES*MOC_RES) + vy*MOC_RES + vz; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mesh_octree_add_node(MocNode **bt, intptr_t index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(*bt==NULL) { | 
					
						
							|  |  |  | 		*bt= MEM_callocN(sizeof(MocNode), "MocNode"); | 
					
						
							|  |  |  | 		(*bt)->index[0]= index; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		int a; | 
					
						
							|  |  |  | 		for(a=0; a<MOC_NODE_RES; a++) { | 
					
						
							|  |  |  | 			if((*bt)->index[a]==index) | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			else if((*bt)->index[a]==0) { | 
					
						
							|  |  |  | 				(*bt)->index[a]= index; | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		mesh_octree_add_node(&(*bt)->next, index); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mesh_octree_free_node(MocNode **bt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if( (*bt)->next ) { | 
					
						
							|  |  |  | 		mesh_octree_free_node(&(*bt)->next); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	MEM_freeN(*bt); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* temporal define, just to make nicer code below */ | 
					
						
							|  |  |  | #define MOC_ADDNODE(vx, vy, vz)	mesh_octree_add_node(basetable + ((vx)*MOC_RES*MOC_RES) + (vy)*MOC_RES + (vz), index)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mesh_octree_add_nodes(MocNode **basetable, float *co, float *offs, float *div, intptr_t index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float fx, fy, fz; | 
					
						
							|  |  |  | 	int vx, vy, vz; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-06-08 20:08:19 +00:00
										 |  |  | 	if (!finite(co[0]) || | 
					
						
							|  |  |  | 		!finite(co[1]) || | 
					
						
							|  |  |  | 		!finite(co[2]) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	fx= (co[0]-offs[0])/div[0]; | 
					
						
							|  |  |  | 	fy= (co[1]-offs[1])/div[1]; | 
					
						
							|  |  |  | 	fz= (co[2]-offs[2])/div[2]; | 
					
						
							|  |  |  | 	CLAMP(fx, 0.0f, MOC_RES-MOC_THRESH); | 
					
						
							|  |  |  | 	CLAMP(fy, 0.0f, MOC_RES-MOC_THRESH); | 
					
						
							|  |  |  | 	CLAMP(fz, 0.0f, MOC_RES-MOC_THRESH); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	vx= floor(fx); | 
					
						
							|  |  |  | 	vy= floor(fy); | 
					
						
							|  |  |  | 	vz= floor(fz); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	MOC_ADDNODE(vx, vy, vz); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if( vx>0 ) | 
					
						
							|  |  |  | 		if( fx-((float)vx)-MOC_THRESH < 0.0f) | 
					
						
							|  |  |  | 			MOC_ADDNODE(vx-1, vy, vz); | 
					
						
							|  |  |  | 	if( vx<MOC_RES-2 ) | 
					
						
							|  |  |  | 		if( fx-((float)vx)+MOC_THRESH > 1.0f) | 
					
						
							|  |  |  | 			MOC_ADDNODE(vx+1, vy, vz); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if( vy>0 ) | 
					
						
							|  |  |  | 		if( fy-((float)vy)-MOC_THRESH < 0.0f)  | 
					
						
							|  |  |  | 			MOC_ADDNODE(vx, vy-1, vz); | 
					
						
							|  |  |  | 	if( vy<MOC_RES-2 ) | 
					
						
							|  |  |  | 		if( fy-((float)vy)+MOC_THRESH > 1.0f)  | 
					
						
							|  |  |  | 			MOC_ADDNODE(vx, vy+1, vz); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if( vz>0 ) | 
					
						
							|  |  |  | 		if( fz-((float)vz)-MOC_THRESH < 0.0f)  | 
					
						
							|  |  |  | 			MOC_ADDNODE(vx, vy, vz-1); | 
					
						
							|  |  |  | 	if( vz<MOC_RES-2 ) | 
					
						
							|  |  |  | 		if( fz-((float)vz)+MOC_THRESH > 1.0f)  | 
					
						
							|  |  |  | 			MOC_ADDNODE(vx, vy, vz+1); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | static intptr_t mesh_octree_find_index(MocNode **bt, MVert *mvert, float *co) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float *vec; | 
					
						
							|  |  |  | 	int a; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(*bt==NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	for(a=0; a<MOC_NODE_RES; a++) { | 
					
						
							|  |  |  | 		if((*bt)->index[a]) { | 
					
						
							|  |  |  | 			/* does mesh verts and editmode, code looks potential dangerous, octree should really be filled OK! */ | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 			if(mvert) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				vec= (mvert+(*bt)->index[a]-1)->co; | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 				if(compare_v3v3(vec, co, MOC_THRESH)) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					return (*bt)->index[a]-1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				EditVert *eve= (EditVert *)((*bt)->index[a]); | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 				if(compare_v3v3(eve->co, co, MOC_THRESH)) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					return (*bt)->index[a]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if( (*bt)->next) | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 		return mesh_octree_find_index(&(*bt)->next, mvert, co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct { | 
					
						
							|  |  |  | 	MocNode **table; | 
					
						
							|  |  |  | 	float offs[3], div[3]; | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | } MeshOctree = {NULL, {0, 0, 0}, {0, 0, 0}}; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* mode is 's' start, or 'e' end, or 'u' use */ | 
					
						
							|  |  |  | /* if end, ob can be NULL */ | 
					
						
							|  |  |  | intptr_t mesh_octree_table(Object *ob, EditMesh *em, float *co, char mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MocNode **bt; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(mode=='u') {		/* use table */ | 
					
						
							|  |  |  | 		if(MeshOctree.table==NULL) | 
					
						
							|  |  |  | 			mesh_octree_table(ob, em, NULL, 's'); | 
					
						
							|  |  |  | 	    | 
					
						
							|  |  |  | 		if(MeshOctree.table) { | 
					
						
							|  |  |  | 			Mesh *me= ob->data; | 
					
						
							|  |  |  | 			bt= MeshOctree.table + mesh_octree_get_base_offs(co, MeshOctree.offs, MeshOctree.div); | 
					
						
							| 
									
										
										
										
											2009-02-23 17:30:00 +00:00
										 |  |  | 			if(em) | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 				return mesh_octree_find_index(bt, NULL, co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			else | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 				return mesh_octree_find_index(bt, me->mvert, co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(mode=='s') {	/* start table */ | 
					
						
							|  |  |  | 		Mesh *me= ob->data; | 
					
						
							|  |  |  | 		float min[3], max[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* we compute own bounding box and don't reuse ob->bb because
 | 
					
						
							|  |  |  | 		 * we are using the undeformed coordinates*/ | 
					
						
							|  |  |  | 		INIT_MINMAX(min, max); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-10 14:19:14 +00:00
										 |  |  | 		if(em && me->edit_mesh==em) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			EditVert *eve; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			for(eve= em->verts.first; eve; eve= eve->next) | 
					
						
							|  |  |  | 				DO_MINMAX(eve->co, min, max) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else {		 | 
					
						
							|  |  |  | 			MVert *mvert; | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 			int a; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 			for(a=0, mvert= me->mvert; a<me->totvert; a++, mvert++) | 
					
						
							|  |  |  | 				DO_MINMAX(mvert->co, min, max); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* for quick unit coordinate calculus */ | 
					
						
							|  |  |  | 		VECCOPY(MeshOctree.offs, min); | 
					
						
							|  |  |  | 		MeshOctree.offs[0]-= MOC_THRESH;		/* we offset it 1 threshold unit extra */ | 
					
						
							|  |  |  | 		MeshOctree.offs[1]-= MOC_THRESH; | 
					
						
							|  |  |  | 		MeshOctree.offs[2]-= MOC_THRESH; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		sub_v3_v3v3(MeshOctree.div, max, min); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		MeshOctree.div[0]+= 2*MOC_THRESH;	/* and divide with 2 threshold unit more extra (try 8x8 unit grid on paint) */ | 
					
						
							|  |  |  | 		MeshOctree.div[1]+= 2*MOC_THRESH; | 
					
						
							|  |  |  | 		MeshOctree.div[2]+= 2*MOC_THRESH; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		mul_v3_fl(MeshOctree.div, 1.0f/MOC_RES); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(MeshOctree.div[0]==0.0f) MeshOctree.div[0]= 1.0f; | 
					
						
							|  |  |  | 		if(MeshOctree.div[1]==0.0f) MeshOctree.div[1]= 1.0f; | 
					
						
							|  |  |  | 		if(MeshOctree.div[2]==0.0f) MeshOctree.div[2]= 1.0f; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 		if(MeshOctree.table) /* happens when entering this call without ending it */ | 
					
						
							|  |  |  | 			mesh_octree_table(ob, em, co, 'e'); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		MeshOctree.table= MEM_callocN(MOC_RES*MOC_RES*MOC_RES*sizeof(void *), "sym table"); | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2009-01-10 14:19:14 +00:00
										 |  |  | 		if(em && me->edit_mesh==em) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			EditVert *eve; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for(eve= em->verts.first; eve; eve= eve->next) { | 
					
						
							|  |  |  | 				mesh_octree_add_nodes(MeshOctree.table, eve->co, MeshOctree.offs, MeshOctree.div, (intptr_t)(eve)); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else {		 | 
					
						
							|  |  |  | 			MVert *mvert; | 
					
						
							|  |  |  | 			int a; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 			for(a=0, mvert= me->mvert; a<me->totvert; a++, mvert++) | 
					
						
							|  |  |  | 				mesh_octree_add_nodes(MeshOctree.table, mvert->co, MeshOctree.offs, MeshOctree.div, a+1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(mode=='e') { /* end table */ | 
					
						
							|  |  |  | 		if(MeshOctree.table) { | 
					
						
							|  |  |  | 			int a; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			for(a=0, bt=MeshOctree.table; a<MOC_RES*MOC_RES*MOC_RES; a++, bt++) { | 
					
						
							|  |  |  | 				if(*bt) mesh_octree_free_node(bt); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			MEM_freeN(MeshOctree.table); | 
					
						
							|  |  |  | 			MeshOctree.table= NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ********************* MESH VERTEX MIRR TOPO LOOKUP *************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MIRRHASH_TYPE int
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct MirrTopoPair { | 
					
						
							|  |  |  | 	long hash; | 
					
						
							|  |  |  | 	int vIndex; | 
					
						
							|  |  |  | } MirrTopoPair; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int MirrTopo_long_sort(const void *l1, const void *l2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(			(MIRRHASH_TYPE)(intptr_t)l1 > (MIRRHASH_TYPE)(intptr_t)l2 ) return  1; | 
					
						
							|  |  |  | 	else if(	(MIRRHASH_TYPE)(intptr_t)l1 < (MIRRHASH_TYPE)(intptr_t)l2 ) return -1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int MirrTopo_item_sort(const void *v1, const void *v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(			((MirrTopoPair *)v1)->hash > ((MirrTopoPair *)v2)->hash ) return  1; | 
					
						
							|  |  |  | 	else if(	((MirrTopoPair *)v1)->hash < ((MirrTopoPair *)v2)->hash ) return -1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static long *mesh_topo_lookup = NULL; | 
					
						
							|  |  |  | static int  mesh_topo_lookup_tot = -1; | 
					
						
							|  |  |  | static int  mesh_topo_lookup_mode = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* mode is 's' start, or 'e' end, or 'u' use */ | 
					
						
							|  |  |  | /* if end, ob can be NULL */ | 
					
						
							|  |  |  | long mesh_mirrtopo_table(Object *ob, char mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(mode=='u') {		/* use table */ | 
					
						
							|  |  |  | 		Mesh *me= ob->data; | 
					
						
							|  |  |  | 		if(	(mesh_topo_lookup==NULL) || | 
					
						
							|  |  |  | 			(mesh_topo_lookup_mode != ob->mode) || | 
					
						
							|  |  |  | 			(me->edit_mesh && me->edit_mesh->totvert != mesh_topo_lookup_tot) || | 
					
						
							|  |  |  | 			(me->edit_mesh==NULL && me->totvert != mesh_topo_lookup_tot) | 
					
						
							|  |  |  | 		) { | 
					
						
							|  |  |  | 			mesh_mirrtopo_table(ob, 's'); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if(mode=='s') { /* start table */ | 
					
						
							|  |  |  | 		Mesh *me= ob->data; | 
					
						
							|  |  |  | 		MEdge *medge; | 
					
						
							|  |  |  | 		EditMesh *em= me->edit_mesh; | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 		void **eve_tmp_back= NULL; /* some of the callers are using eve->tmp so restore after */ | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* editmode*/ | 
					
						
							|  |  |  | 		EditEdge *eed; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		int a, last, totvert; | 
					
						
							|  |  |  | 		int totUnique= -1, totUniqueOld= -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MIRRHASH_TYPE *MirrTopoHash = NULL; | 
					
						
							|  |  |  | 		MIRRHASH_TYPE *MirrTopoHash_Prev = NULL; | 
					
						
							|  |  |  | 		MirrTopoPair *MirrTopoPairs; | 
					
						
							| 
									
										
										
										
											2010-02-17 23:04:30 +00:00
										 |  |  | 		mesh_topo_lookup_mode= ob->mode; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* reallocate if needed */ | 
					
						
							|  |  |  | 		if (mesh_topo_lookup) { | 
					
						
							|  |  |  | 			MEM_freeN(mesh_topo_lookup); | 
					
						
							|  |  |  | 			mesh_topo_lookup = NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(em) { | 
					
						
							|  |  |  | 			EditVert *eve; | 
					
						
							|  |  |  | 			totvert= 0; | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 			eve_tmp_back=  MEM_callocN( em->totvert * sizeof(void *), "TopoMirr" ); | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 			for(eve= em->verts.first; eve; eve= eve->next) { | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 				eve_tmp_back[totvert]= eve->tmp.p; | 
					
						
							|  |  |  | 				eve->tmp.l = totvert++; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 			totvert = me->totvert; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MirrTopoHash = MEM_callocN( totvert * sizeof(MIRRHASH_TYPE), "TopoMirr" ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Initialize the vert-edge-user counts used to detect unique topology */ | 
					
						
							|  |  |  | 		if(em) { | 
					
						
							|  |  |  | 			for(eed=em->edges.first; eed; eed= eed->next) { | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 				MirrTopoHash[eed->v1->tmp.l]++; | 
					
						
							|  |  |  | 				MirrTopoHash[eed->v2->tmp.l]++; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			for(a=0, medge=me->medge; a<me->totedge; a++, medge++) { | 
					
						
							|  |  |  | 				MirrTopoHash[medge->v1]++; | 
					
						
							|  |  |  | 				MirrTopoHash[medge->v2]++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MirrTopoHash_Prev = MEM_dupallocN( MirrTopoHash ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		totUniqueOld = -1; | 
					
						
							|  |  |  | 		while(1) { | 
					
						
							|  |  |  | 			/* use the number of edges per vert to give verts unique topology IDs */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(em) { | 
					
						
							|  |  |  | 				for(eed=em->edges.first; eed; eed= eed->next) { | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 					MirrTopoHash[eed->v1->tmp.l] += MirrTopoHash_Prev[eed->v2->tmp.l]; | 
					
						
							|  |  |  | 					MirrTopoHash[eed->v2->tmp.l] += MirrTopoHash_Prev[eed->v1->tmp.l]; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				for(a=0, medge=me->medge; a<me->totedge; a++, medge++) { | 
					
						
							|  |  |  | 					/* This can make realy big numbers, wrapping around here is fine */ | 
					
						
							|  |  |  | 					MirrTopoHash[medge->v1] += MirrTopoHash_Prev[medge->v2]; | 
					
						
							|  |  |  | 					MirrTopoHash[medge->v2] += MirrTopoHash_Prev[medge->v1]; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			memcpy(MirrTopoHash_Prev, MirrTopoHash, sizeof(MIRRHASH_TYPE) * totvert); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* sort so we can count unique values */ | 
					
						
							|  |  |  | 			qsort(MirrTopoHash_Prev, totvert, sizeof(MIRRHASH_TYPE), MirrTopo_long_sort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			totUnique = 1; /* account for skiping the first value */ | 
					
						
							|  |  |  | 			for(a=1; a<totvert; a++) { | 
					
						
							|  |  |  | 				if (MirrTopoHash_Prev[a-1] != MirrTopoHash_Prev[a]) { | 
					
						
							|  |  |  | 					totUnique++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (totUnique <= totUniqueOld) { | 
					
						
							|  |  |  | 				/* Finish searching for unique valus when 1 loop dosnt give a
 | 
					
						
							|  |  |  | 				 * higher number of unique values compared to the previous loop */ | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				totUniqueOld = totUnique; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			/* Copy the hash calculated this iter, so we can use them next time */ | 
					
						
							|  |  |  | 			memcpy(MirrTopoHash_Prev, MirrTopoHash, sizeof(MIRRHASH_TYPE) * totvert); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 		/* restore eve->tmp.* */ | 
					
						
							|  |  |  | 		if(eve_tmp_back) { | 
					
						
							|  |  |  | 			EditVert *eve; | 
					
						
							|  |  |  | 			totvert= 0; | 
					
						
							|  |  |  | 			for(eve= em->verts.first; eve; eve= eve->next) { | 
					
						
							|  |  |  | 				eve->tmp.p= eve_tmp_back[totvert++]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			MEM_freeN(eve_tmp_back); | 
					
						
							|  |  |  | 			eve_tmp_back= NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		/* Hash/Index pairs are needed for sorting to find index pairs */ | 
					
						
							|  |  |  | 		MirrTopoPairs= MEM_callocN( sizeof(MirrTopoPair) * totvert, "MirrTopoPairs"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* since we are looping through verts, initialize these values here too */ | 
					
						
							|  |  |  | 		mesh_topo_lookup = MEM_mallocN( totvert * sizeof(long), "mesh_topo_lookup" ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(em) { | 
					
						
							|  |  |  | 			EM_init_index_arrays(em,1,0,0); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for(a=0; a<totvert; a++) { | 
					
						
							|  |  |  | 			MirrTopoPairs[a].hash= MirrTopoHash[a]; | 
					
						
							|  |  |  | 			MirrTopoPairs[a].vIndex = a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* initialize lookup */ | 
					
						
							|  |  |  | 			mesh_topo_lookup[a] = -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		qsort(MirrTopoPairs, totvert, sizeof(MirrTopoPair), MirrTopo_item_sort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Since the loop starts at 2, we must define the last index where the hash's differ */ | 
					
						
							|  |  |  | 		last = ((totvert >= 2) && (MirrTopoPairs[0].hash == MirrTopoPairs[1].hash)) ? 0 : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Get the pairs out of the sorted hashes, note, totvert+1 means we can use the previous 2,
 | 
					
						
							|  |  |  | 		 * but you cant ever access the last 'a' index of MirrTopoPairs */ | 
					
						
							|  |  |  | 		for(a=2; a < totvert+1; a++) { | 
					
						
							|  |  |  | 			/* printf("I %d %ld %d\n", (a-last), MirrTopoPairs[a  ].hash, MirrTopoPairs[a  ].vIndex ); */ | 
					
						
							|  |  |  | 			if ((a==totvert) || (MirrTopoPairs[a-1].hash != MirrTopoPairs[a].hash)) { | 
					
						
							|  |  |  | 				if (a-last==2) { | 
					
						
							|  |  |  | 					if(em) { | 
					
						
							|  |  |  | 						mesh_topo_lookup[MirrTopoPairs[a-1].vIndex] =	(long)EM_get_vert_for_index(MirrTopoPairs[a-2].vIndex); | 
					
						
							|  |  |  | 						mesh_topo_lookup[MirrTopoPairs[a-2].vIndex] =	(long)EM_get_vert_for_index(MirrTopoPairs[a-1].vIndex); | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						mesh_topo_lookup[MirrTopoPairs[a-1].vIndex] =	MirrTopoPairs[a-2].vIndex; | 
					
						
							|  |  |  | 						mesh_topo_lookup[MirrTopoPairs[a-2].vIndex] =	MirrTopoPairs[a-1].vIndex; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				last= a; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(em) { | 
					
						
							|  |  |  | 			EM_free_index_arrays(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MEM_freeN( MirrTopoPairs ); | 
					
						
							|  |  |  | 		MirrTopoPairs = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MEM_freeN( MirrTopoHash ); | 
					
						
							|  |  |  | 		MEM_freeN( MirrTopoHash_Prev ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		mesh_topo_lookup_tot = totvert; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} else if(mode=='e') { /* end table */ | 
					
						
							|  |  |  | 		if (mesh_topo_lookup) { | 
					
						
							|  |  |  | 			MEM_freeN(mesh_topo_lookup); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		mesh_topo_lookup = NULL; | 
					
						
							|  |  |  | 		mesh_topo_lookup_tot= -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int mesh_get_x_mirror_vert_spacial(Object *ob, int index) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Mesh *me= ob->data; | 
					
						
							|  |  |  | 	MVert *mvert; | 
					
						
							|  |  |  | 	float vec[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-10-22 09:31:07 +00:00
										 |  |  | 	mvert= me->mvert+index; | 
					
						
							|  |  |  | 	vec[0]= -mvert->co[0]; | 
					
						
							|  |  |  | 	vec[1]= mvert->co[1]; | 
					
						
							|  |  |  | 	vec[2]= mvert->co[2]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	return mesh_octree_table(ob, NULL, vec, 'u'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | static int mesh_get_x_mirror_vert_topo(Object *ob, int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (mesh_mirrtopo_table(ob, 'u')==-1) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return mesh_topo_lookup[index]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int mesh_get_x_mirror_vert(Object *ob, int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (((Mesh *)ob->data)->editflag & ME_EDIT_MIRROR_TOPO) { | 
					
						
							|  |  |  | 		return mesh_get_x_mirror_vert_topo(ob, index); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return mesh_get_x_mirror_vert_spacial(ob, index); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static EditVert *editmesh_get_x_mirror_vert_spacial(Object *ob, EditMesh *em, float *co) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec[3]; | 
					
						
							|  |  |  | 	intptr_t poinval; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* ignore nan verts */ | 
					
						
							| 
									
										
										
										
											2009-06-08 20:08:19 +00:00
										 |  |  | 	if (!finite(co[0]) || | 
					
						
							|  |  |  | 		!finite(co[1]) || | 
					
						
							|  |  |  | 		!finite(co[2]) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	   ) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	vec[0]= -co[0]; | 
					
						
							|  |  |  | 	vec[1]= co[1]; | 
					
						
							|  |  |  | 	vec[2]= co[2]; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	poinval= mesh_octree_table(ob, em, vec, 'u'); | 
					
						
							|  |  |  | 	if(poinval != -1) | 
					
						
							|  |  |  | 		return (EditVert *)(poinval); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | static EditVert *editmesh_get_x_mirror_vert_topo(Object *ob, struct EditMesh *em, EditVert *eve, int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	long poinval; | 
					
						
							|  |  |  | 	if (mesh_mirrtopo_table(ob, 'u')==-1) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 	if (index == -1) { | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		index = BLI_findindex(&em->verts, eve); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 		if (index == -1) { | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	poinval= mesh_topo_lookup[ index ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(poinval != -1) | 
					
						
							|  |  |  | 		return (EditVert *)(poinval); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2010-10-16 02:40:31 +00:00
										 |  |  | }	 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | EditVert *editmesh_get_x_mirror_vert(Object *ob, struct EditMesh *em, EditVert *eve, float *co, int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (((Mesh *)ob->data)->editflag & ME_EDIT_MIRROR_TOPO) { | 
					
						
							|  |  |  | 		return editmesh_get_x_mirror_vert_topo(ob, em, eve, index); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2010-10-16 02:40:31 +00:00
										 |  |  | 		return editmesh_get_x_mirror_vert_spacial(ob, em, co); | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | float *editmesh_get_mirror_uv(int axis, float *uv, float *mirrCent, float *face_cent) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float vec[2]; | 
					
						
							|  |  |  | 	float cent_vec[2]; | 
					
						
							|  |  |  | 	float cent[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ignore nan verts */ | 
					
						
							|  |  |  | 	if (isnan(uv[0]) || !finite(uv[0]) || | 
					
						
							|  |  |  | 		isnan(uv[1]) || !finite(uv[1]) | 
					
						
							|  |  |  | 	   ) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (axis) { | 
					
						
							|  |  |  | 		vec[0]= uv[0]; | 
					
						
							|  |  |  | 		vec[1]= -((uv[1])-mirrCent[1]) + mirrCent[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cent_vec[0] = face_cent[0]; | 
					
						
							|  |  |  | 		cent_vec[1]= -((face_cent[1])-mirrCent[1]) + mirrCent[1]; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		vec[0]= -((uv[0])-mirrCent[0]) + mirrCent[0]; | 
					
						
							|  |  |  | 		vec[1]= uv[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cent_vec[0]= -((face_cent[0])-mirrCent[0]) + mirrCent[0]; | 
					
						
							|  |  |  | 		cent_vec[1] = face_cent[1]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* TODO - Optimize */ | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		EditFace *efa; | 
					
						
							|  |  |  | 		int i, len; | 
					
						
							|  |  |  | 		for(efa=em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 			MTFace *tf= (MTFace *)CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							|  |  |  | 			uv_center(tf->uv, cent, (void *)efa->v4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if ( (fabs(cent[0] - cent_vec[0]) < 0.001) && (fabs(cent[1] - cent_vec[1]) < 0.001) ) { | 
					
						
							|  |  |  | 				len = efa->v4 ? 4 : 3; | 
					
						
							|  |  |  | 				for (i=0; i<len; i++) { | 
					
						
							|  |  |  | 					if ( (fabs(tf->uv[i][0] - vec[0]) < 0.001) && (fabs(tf->uv[i][1] - vec[1]) < 0.001) ) { | 
					
						
							|  |  |  | 						return tf->uv[i]; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | static unsigned int mirror_facehash(void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MFace *mf= ptr; | 
					
						
							|  |  |  | 	int v0, v1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(mf->v4) { | 
					
						
							|  |  |  | 		v0= MIN4(mf->v1, mf->v2, mf->v3, mf->v4); | 
					
						
							|  |  |  | 		v1= MAX4(mf->v1, mf->v2, mf->v3, mf->v4); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		v0= MIN3(mf->v1, mf->v2, mf->v3); | 
					
						
							|  |  |  | 		v1= MAX3(mf->v1, mf->v2, mf->v3); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ((v0*39)^(v1*31)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int mirror_facerotation(MFace *a, MFace *b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(b->v4) { | 
					
						
							|  |  |  | 		if(a->v1==b->v1 && a->v2==b->v2 && a->v3==b->v3 && a->v4==b->v4) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		else if(a->v4==b->v1 && a->v1==b->v2 && a->v2==b->v3 && a->v3==b->v4) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		else if(a->v3==b->v1 && a->v4==b->v2 && a->v1==b->v3 && a->v2==b->v4) | 
					
						
							|  |  |  | 			return 2; | 
					
						
							|  |  |  | 		else if(a->v2==b->v1 && a->v3==b->v2 && a->v4==b->v3 && a->v1==b->v4) | 
					
						
							|  |  |  | 			return 3; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		if(a->v1==b->v1 && a->v2==b->v2 && a->v3==b->v3) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		else if(a->v3==b->v1 && a->v1==b->v2 && a->v2==b->v3) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		else if(a->v2==b->v1 && a->v3==b->v2 && a->v1==b->v3) | 
					
						
							|  |  |  | 			return 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int mirror_facecmp(void *a, void *b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (mirror_facerotation((MFace*)a, (MFace*)b) == -1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int *mesh_get_x_mirror_faces(Object *ob, EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Mesh *me= ob->data; | 
					
						
							|  |  |  | 	MVert *mv, *mvert= me->mvert; | 
					
						
							|  |  |  | 	MFace mirrormf, *mf, *hashmf, *mface= me->mface; | 
					
						
							|  |  |  | 	GHash *fhash; | 
					
						
							|  |  |  | 	int *mirrorverts, *mirrorfaces; | 
					
						
							|  |  |  | 	int a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mirrorverts= MEM_callocN(sizeof(int)*me->totvert, "MirrorVerts"); | 
					
						
							|  |  |  | 	mirrorfaces= MEM_callocN(sizeof(int)*2*me->totface, "MirrorFaces"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mesh_octree_table(ob, em, NULL, 's'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(a=0, mv=mvert; a<me->totvert; a++, mv++) | 
					
						
							|  |  |  | 		mirrorverts[a]= mesh_get_x_mirror_vert(ob, a); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mesh_octree_table(ob, em, NULL, 'e'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-07 07:54:25 +00:00
										 |  |  | 	fhash= BLI_ghash_new(mirror_facehash, mirror_facecmp, "mirror_facehash gh"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(a=0, mf=mface; a<me->totface; a++, mf++) | 
					
						
							|  |  |  | 		BLI_ghash_insert(fhash, mf, mf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(a=0, mf=mface; a<me->totface; a++, mf++) { | 
					
						
							|  |  |  | 		mirrormf.v1= mirrorverts[mf->v3]; | 
					
						
							|  |  |  | 		mirrormf.v2= mirrorverts[mf->v2]; | 
					
						
							|  |  |  | 		mirrormf.v3= mirrorverts[mf->v1]; | 
					
						
							|  |  |  | 		mirrormf.v4= (mf->v4)? mirrorverts[mf->v4]: 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* make sure v4 is not 0 if a quad */ | 
					
						
							|  |  |  | 		if(mf->v4 && mirrormf.v4==0) { | 
					
						
							|  |  |  | 			SWAP(int, mirrormf.v1, mirrormf.v3); | 
					
						
							|  |  |  | 			SWAP(int, mirrormf.v2, mirrormf.v4); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		hashmf= BLI_ghash_lookup(fhash, &mirrormf); | 
					
						
							|  |  |  | 		if(hashmf) { | 
					
						
							|  |  |  | 			mirrorfaces[a*2]= hashmf - mface; | 
					
						
							|  |  |  | 			mirrorfaces[a*2+1]= mirror_facerotation(&mirrormf, hashmf); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			mirrorfaces[a*2]= -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_ghash_free(fhash, NULL, NULL); | 
					
						
							|  |  |  | 	MEM_freeN(mirrorverts); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return mirrorfaces; | 
					
						
							|  |  |  | } |