| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											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 ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:29:51 +00:00
										 |  |  | /** \file blender/editors/mesh/meshtools.c
 | 
					
						
							|  |  |  |  *  \ingroup edmesh | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2012-03-03 16:31:46 +00:00
										 |  |  |  * meshtools.c: no editmode (violated already :), tools operating on meshes | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2009-09-16 17:43:09 +00:00
										 |  |  | #include <float.h>
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 22:17:30 +00:00
										 |  |  | #include "DNA_mesh_types.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"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #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"
 | 
					
						
							| 
									
										
										
										
											2009-05-16 16:18:08 +00:00
										 |  |  | #include "BKE_tessmesh.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"
 | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | #include "uvedit_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
 | 
					
						
							| 
									
										
										
										
											2012-03-03 16:31:46 +00:00
										 |  |  |  * return 0 if no join is made (error) and 1 if the join is done */ | 
					
						
							| 
									
										
										
										
											2009-07-11 10:20:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | int join_mesh_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  | 	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; | 
					
						
							| 
									
										
										
										
											2011-01-13 04:53:55 +00:00
										 |  |  | 	MVert *mvert, *mv; | 
					
						
							|  |  |  | 	MEdge *medge = NULL; | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 	MPoly *mpoly = NULL; | 
					
						
							|  |  |  | 	MLoop *mloop = NULL; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	Key *key, *nkey = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	KeyBlock *kb, *okb, *kbn; | 
					
						
							|  |  |  | 	float imat[4][4], cmat[4][4], *fp1, *fp2, curpos; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	int a, b, totcol, totmat = 0, totedge = 0, totvert = 0, ok = 0; | 
					
						
							|  |  |  | 	int totloop = 0, totpoly = 0, vertofs, *matmap = NULL; | 
					
						
							|  |  |  | 	int i, j, index, haskey = 0, edgeofs, loopofs, polyofs; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	bDeformGroup *dg, *odg; | 
					
						
							|  |  |  | 	MDeformVert *dvert; | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 	CustomData vdata, edata, fdata, ldata, pdata; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (scene->obedit) { | 
					
						
							| 
									
										
										
										
											2010-12-21 15:10:09 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_WARNING, "Cant join while in editmode"); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (!ob || ob->type != OB_MESH) { | 
					
						
							| 
									
										
										
										
											2010-12-21 15:10:09 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_WARNING, "Active object is not a mesh"); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* count & check */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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; | 
					
						
							|  |  |  | 			totloop += me->totloop; | 
					
						
							|  |  |  | 			totpoly += me->totpoly; | 
					
						
							|  |  |  | 			totmat += base->object->totcol; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (base->object == ob) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				ok = 1; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			/* check for shapekeys */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (me->key) | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				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 */  | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (ok == 0) { | 
					
						
							| 
									
										
										
										
											2010-12-21 15:10:09 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_WARNING, "Active object is not a selected mesh"); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	me = (Mesh *)ob->data; | 
					
						
							|  |  |  | 	key = me->key; | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (totvert == 0 || totvert == me->totvert) { | 
					
						
							| 
									
										
										
										
											2010-12-21 15:10:09 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_WARNING, "No mesh data to join"); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (totvert > MESH_MAX_VERTS) { | 
					
						
							| 
									
										
										
										
											2010-12-21 15:10:09 +00:00
										 |  |  | 		BKE_reportf(op->reports, RPT_WARNING, "Joining results in %d vertices, limit is " STRINGIFY(MESH_MAX_VERTS), totvert); | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 		return OPERATOR_CANCELLED;		 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* new material indices and material array */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	matar = MEM_callocN(sizeof(void *) * totmat, "join_mesh matar"); | 
					
						
							|  |  |  | 	if (totmat) matmap = MEM_callocN(sizeof(int) * totmat, "join_mesh matmap"); | 
					
						
							|  |  |  | 	totcol = ob->totcol; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* obact materials in new main array, is nicer start! */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	for (a = 0; a < ob->totcol; a++) { | 
					
						
							|  |  |  | 		matar[a] = give_current_material(ob, a + 1); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		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
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	 *  with arrays that are large enough to hold shapekey data for all meshes | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	 * -	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 | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (key) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		/* make a duplicate copy that will only be used here... (must remember to free it!) */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		nkey = copy_key(key); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* for all keys in old block, clear data-arrays */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		for (kb = key->block.first; kb; kb = kb->next) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (kb->data) MEM_freeN(kb->data); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			kb->data = MEM_callocN(sizeof(float) * 3 * totvert, "join_shapekey"); | 
					
						
							|  |  |  | 			kb->totelem = totvert; | 
					
						
							|  |  |  | 			kb->weights = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	else if (haskey) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		/* add a new key-block and add to the mesh */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		key = me->key = add_key((ID *)me); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		key->type = KEY_RELATIVE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* first pass over objects - copying materials and vertexgroups across */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	CTX_DATA_BEGIN(C, Base *, base, selected_editable_bases) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		/* only act if a mesh, and not the one we're joining to */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if ((ob != base->object) && (base->object->type == OB_MESH)) { | 
					
						
							|  |  |  | 			me = base->object->data; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			/* Join this object's vertex groups to the base one's */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			for (dg = base->object->defbase.first; dg; dg = dg->next) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* See if this group exists in the object (if it doesn't, add it to the end) */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +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); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			if (ob->defbase.first && ob->actdef == 0) | 
					
						
							|  |  |  | 				ob->actdef = 1; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (me->totvert) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* Add this object's materials to the base one's if they don't exist already (but only if limits not exceeded yet) */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 				if (totcol < MAXMAT) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 						for (b = 0; b < totcol; b++) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (ma == matar[b]) break; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 						} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 						if (b == totcol) { | 
					
						
							|  |  |  | 							matar[b] = ma; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (ma) { | 
					
						
							| 
									
										
										
										
											2011-04-26 07:17:21 +00:00
										 |  |  | 								id_us_plus(&ma->id); | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							totcol++; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 						if (totcol >= MAXMAT) | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				/* if this mesh has shapekeys, check if destination mesh already has matching entries too */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 				if (me->key && key) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					for (kb = me->key->block.first; kb; kb = kb->next) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 						/* if key doesn't exist in destination mesh, add it */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 						if (key_get_named_keyblock(key, kb->name) == NULL) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							/* copy this existing one over to the new shapekey block */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							kbn = MEM_dupallocN(kb); | 
					
						
							|  |  |  | 							kbn->prev = kbn->next = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							 | 
					
						
							| 
									
										
										
										
											2012-04-12 11:50:43 +00:00
										 |  |  | 							/* adjust settings to fit (allocate a new data-array) */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							kbn->data = MEM_callocN(sizeof(float) * 3 * totvert, "joined_shapekey"); | 
					
						
							|  |  |  | 							kbn->totelem = totvert; | 
					
						
							|  |  |  | 							kbn->weights = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							okb = key->block.last; | 
					
						
							|  |  |  | 							curpos = (okb) ? okb->pos : -0.1f; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (key->type == KEY_RELATIVE) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								kbn->pos = curpos + 0.1f; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							else | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								kbn->pos = curpos; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							 | 
					
						
							|  |  |  | 							BLI_addtail(&key->block, kbn); | 
					
						
							|  |  |  | 							key->totkey++; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							if (key->totkey == 1) key->refkey = kbn; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							 | 
					
						
							|  |  |  | 							// XXX 2.5 Animato
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 							/* also, copy corresponding ipo-curve to ipo-block if applicable */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (me->key->ipo && key->ipo) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 								// 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)); | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 	memset(&ldata, 0, sizeof(ldata)); | 
					
						
							|  |  |  | 	memset(&pdata, 0, sizeof(pdata)); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	mvert = CustomData_add_layer(&vdata, CD_MVERT, CD_CALLOC, NULL, totvert); | 
					
						
							|  |  |  | 	medge = CustomData_add_layer(&edata, CD_MEDGE, CD_CALLOC, NULL, totedge); | 
					
						
							|  |  |  | 	mloop = CustomData_add_layer(&ldata, CD_MLOOP, CD_CALLOC, NULL, totloop); | 
					
						
							|  |  |  | 	mpoly = CustomData_add_layer(&pdata, CD_MPOLY, CD_CALLOC, NULL, totpoly); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vertofs = 0; | 
					
						
							|  |  |  | 	edgeofs = 0; | 
					
						
							|  |  |  | 	loopofs = 0; | 
					
						
							|  |  |  | 	polyofs = 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
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	CTX_DATA_BEGIN(C, Base *, base, selected_editable_bases) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		/* only join if this is a mesh */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (base->object->type == OB_MESH) { | 
					
						
							|  |  |  | 			me = base->object->data; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (me->totvert) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				/* 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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				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 */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 				if (dvert) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					for (i = 0; i < me->totvert; i++) { | 
					
						
							|  |  |  | 						for (j = 0; j < dvert[i].totweight; j++) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							/*	Find the old vertex group */ | 
					
						
							|  |  |  | 							odg = BLI_findlink(&base->object->defbase, dvert[i].dw[j].def_nr); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (odg) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 								/*	Search for a match in the new object, and set new index */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								for (dg = ob->defbase.first, index = 0; dg; dg = dg->next, index++) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 									if (!strcmp(dg->name, odg->name)) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 										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 */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 				if (base->object != ob) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 					/* watch this: switch matmul order really goes wrong */ | 
					
						
							| 
									
										
										
											
												Math lib: matrix multiplication order fix for two functions that were
inconsistent with similar functions & math notation:
mul_m4_m4m4(R, B, A) => mult_m4_m4m4(R, A, B)
mul_m3_m3m4(R, B, A) => mult_m3_m3m4(R, A, B)
For branch maintainers, it should be relatively simple to fix things manually,
it's also possible run this script after merging to do automatic replacement:
http://www.pasteall.org/27459/python
											
										 
											2011-12-16 19:53:12 +00:00
										 |  |  | 					mult_m4_m4m4(cmat, imat, base->object->obmat); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 					 | 
					
						
							|  |  |  | 					/* transform vertex coordinates into new space */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					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) | 
					
						
							|  |  |  | 					 */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 					if (key) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 						/* if this mesh has any shapekeys, check first, otherwise just copy coordinates */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 						for (kb = key->block.first; kb; kb = kb->next) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							/* get pointer to where to write data for this mesh in shapekey's data array */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							fp1 = ((float *)kb->data) + (vertofs * 3); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							 | 
					
						
							|  |  |  | 							/* check if this mesh has such a shapekey */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							okb = key_get_named_keyblock(me->key, kb->name); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (okb) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 								/* copy this mesh's shapekey to the destination shapekey (need to transform first) */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								fp2 = ((float *)(okb->data)); | 
					
						
							|  |  |  | 								for (a = 0; a < me->totvert; a++, fp1 += 3, fp2 += 3) { | 
					
						
							| 
									
										
										
										
											2012-03-09 06:04:17 +00:00
										 |  |  | 									copy_v3_v3(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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								mv = mvert; | 
					
						
							|  |  |  | 								for (a = 0; a < me->totvert; a++, fp1 += 3, mv++) { | 
					
						
							| 
									
										
										
										
											2012-03-09 06:04:17 +00:00
										 |  |  | 									copy_v3_v3(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) | 
					
						
							|  |  |  | 					 */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 					if (key) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 						for (kb = key->block.first; kb; kb = kb->next) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							/* get pointer to where to write data for this mesh in shapekey's data array */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							fp1 = ((float *)kb->data) + (vertofs * 3); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 							 | 
					
						
							|  |  |  | 							/* check if this was one of the original shapekeys */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							okb = key_get_named_keyblock(nkey, kb->name); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 							if (okb) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 								/* copy this mesh's shapekey to the destination shapekey */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								fp2 = ((float *)(okb->data)); | 
					
						
							|  |  |  | 								for (a = 0; a < me->totvert; a++, fp1 += 3, fp2 += 3) { | 
					
						
							| 
									
										
										
										
											2012-03-09 06:04:17 +00:00
										 |  |  | 									copy_v3_v3(fp1, fp2); | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								/* copy base-coordinates to the destination shapekey */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 								mv = mvert; | 
					
						
							|  |  |  | 								for (a = 0; a < me->totvert; a++, fp1 += 3, mv++) { | 
					
						
							| 
									
										
										
										
											2012-03-09 06:04:17 +00:00
										 |  |  | 									copy_v3_v3(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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				mvert += me->totvert; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (me->totedge) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				CustomData_merge(&me->edata, &edata, CD_MASK_MESH, CD_DEFAULT, totedge); | 
					
						
							|  |  |  | 				CustomData_copy_data(&me->edata, &edata, 0, edgeofs, me->totedge); | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				for (a = 0; a < me->totedge; a++, medge++) { | 
					
						
							|  |  |  | 					medge->v1 += vertofs; | 
					
						
							|  |  |  | 					medge->v2 += vertofs; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 			if (me->totloop) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				if (base->object != ob) | 
					
						
							| 
									
										
										
										
											2011-03-31 00:52:12 +00:00
										 |  |  | 					multiresModifier_prepare_join(scene, base->object, ob); | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 				CustomData_merge(&me->ldata, &ldata, CD_MASK_MESH, CD_DEFAULT, totloop); | 
					
						
							|  |  |  | 				CustomData_copy_data(&me->ldata, &ldata, 0, loopofs, me->totloop); | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				for (a = 0; a < me->totloop; a++, mloop++) { | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 					mloop->v += vertofs; | 
					
						
							|  |  |  | 					mloop->e += edgeofs; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (me->totpoly) { | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 				/* make mapping for materials */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				for (a = 1; a <= base->object->totcol; a++) { | 
					
						
							|  |  |  | 					ma = give_current_material(base->object, a); | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					for (b = 0; b < totcol; b++) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 						if (ma == matar[b]) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 							matmap[a - 1] = b; | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 							break; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 				CustomData_merge(&me->pdata, &pdata, CD_MASK_MESH, CD_DEFAULT, totpoly); | 
					
						
							|  |  |  | 				CustomData_copy_data(&me->pdata, &pdata, 0, polyofs, me->totpoly); | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				for (a = 0; a < me->totpoly; a++, mpoly++) { | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 					mpoly->loopstart += loopofs; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					mpoly->mat_nr = matmap ? matmap[(int)mpoly->mat_nr] : 0; | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2011-02-23 00:01:50 +00:00
										 |  |  | 				polyofs += me->totpoly; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-23 00:01:50 +00:00
										 |  |  | 			/* these are used for relinking (cannot be set earlier, 
 | 
					
						
							|  |  |  | 			 * or else reattaching goes wrong) | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 */ | 
					
						
							|  |  |  | 			vertofs += me->totvert; | 
					
						
							| 
									
										
										
										
											2011-02-23 00:01:50 +00:00
										 |  |  | 			edgeofs += me->totedge; | 
					
						
							|  |  |  | 			loopofs += me->totloop; | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			/* free base, now that data is merged */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	me = ob->data; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	CustomData_free(&me->vdata, me->totvert); | 
					
						
							|  |  |  | 	CustomData_free(&me->edata, me->totedge); | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 	CustomData_free(&me->ldata, me->totloop); | 
					
						
							|  |  |  | 	CustomData_free(&me->pdata, me->totpoly); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	me->totvert = totvert; | 
					
						
							|  |  |  | 	me->totedge = totedge; | 
					
						
							|  |  |  | 	me->totloop = totloop; | 
					
						
							|  |  |  | 	me->totpoly = totpoly; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me->vdata = vdata; | 
					
						
							|  |  |  | 	me->edata = edata; | 
					
						
							|  |  |  | 	me->ldata = ldata; | 
					
						
							|  |  |  | 	me->pdata = pdata; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | 	mesh_update_customdata_pointers(me, TRUE); /* BMESH_TODO, check if this arg can be failse, non urgent - campbell */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* old material array */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	for (a = 1; a <= ob->totcol; a++) { | 
					
						
							|  |  |  | 		ma = ob->mat[a - 1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (ma) ma->id.us--; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	for (a = 1; a <= me->totcol; a++) { | 
					
						
							|  |  |  | 		ma = me->mat[a - 1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (ma) ma->id.us--; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (ob->mat) MEM_freeN(ob->mat); | 
					
						
							|  |  |  | 	if (ob->matbits) MEM_freeN(ob->matbits); | 
					
						
							|  |  |  | 	if (me->mat) MEM_freeN(me->mat); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	ob->mat = me->mat = NULL; | 
					
						
							|  |  |  | 	ob->matbits = NULL; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (totcol) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		me->mat = matar; | 
					
						
							|  |  |  | 		ob->mat = MEM_callocN(sizeof(void *) * totcol, "join obmatar"); | 
					
						
							|  |  |  | 		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
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	ob->totcol = me->totcol = totcol; | 
					
						
							| 
									
										
										
										
											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) */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (nkey) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 		// XXX 2.5 Animato
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 		/* free it's ipo too - both are not actually freed from memory yet as ID-blocks */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (nkey->ipo) { | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 			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
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	DAG_scene_sort(bmain, scene);   // removed objects, need to rebuild dag before editmode call
 | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2009-07-13 00:40:20 +00:00
										 |  |  | 	ED_object_enter_editmode(C, EM_WAITCURSOR); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	ED_object_exit_editmode(C, EM_FREEDATA | EM_WAITCURSOR | EM_DO_UNDO); | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	/* toggle editmode using lower level functions so this can be called from python */ | 
					
						
							| 
									
										
										
										
											2012-03-27 04:46:52 +00:00
										 |  |  | 	EDBM_mesh_make(scene->toolsettings, scene, ob); | 
					
						
							| 
									
										
										
										
											2012-03-29 01:41:56 +00:00
										 |  |  | 	EDBM_mesh_load(ob); | 
					
						
							| 
									
										
										
										
											2012-03-27 04:46:52 +00:00
										 |  |  | 	EDBM_mesh_free(me->edit_btmesh); | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 	MEM_freeN(me->edit_btmesh); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	me->edit_btmesh = NULL; | 
					
						
							|  |  |  | 	DAG_id_tag_update(&ob->id, OB_RECALC_OB | OB_RECALC_DATA); | 
					
						
							| 
									
										
										
										
											2010-11-10 01:40:24 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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, 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 16:31:46 +00:00
										 |  |  |  * return 0 if no join is made (error) and 1 of the join is done */ | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int join_mesh_shapes_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	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; | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 	KeyBlock *kb; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	int ok = 0, nonequal_verts = 0; | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	CTX_DATA_BEGIN(C, Base *, base, selected_editable_bases) { | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 		if (base->object == ob) continue; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (base->object->type == OB_MESH) { | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 			selme = (Mesh *)base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			if (selme->totvert == me->totvert) | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 				ok++; | 
					
						
							|  |  |  | 			else | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				nonequal_verts = 1; | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	CTX_DATA_END; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (!ok) { | 
					
						
							|  |  |  | 		if (nonequal_verts) | 
					
						
							| 
									
										
										
										
											2011-09-19 12:26:20 +00:00
										 |  |  | 			BKE_report(op->reports, RPT_WARNING, "Selected meshes must have equal numbers of vertices"); | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2011-09-19 12:26:20 +00:00
										 |  |  | 			BKE_report(op->reports, RPT_WARNING, "No additional selected meshes with equal vertex count to join"); | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (key == NULL) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		key = me->key = add_key((ID *)me); | 
					
						
							|  |  |  | 		key->type = KEY_RELATIVE; | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 16:05:54 +00:00
										 |  |  | 		/* first key added, so it was the basis. initialize it with the existing mesh */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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 */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	CTX_DATA_BEGIN(C, Base *, base, selected_editable_bases) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 		if (base->object == ob) continue; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (base->object->type == OB_MESH) { | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 			selme = (Mesh *)base->object->data; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			if (selme->totvert == me->totvert) { | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 				dm = mesh_get_derived_deform(scene, base->object, CD_MASK_BAREMESH); | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (!dm) continue; | 
					
						
							|  |  |  | 					 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_SCENE | ND_OB_ACTIVE, scene); | 
					
						
							| 
									
										
										
										
											2009-11-28 04:04:01 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	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 */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | #define MOC_RES         8
 | 
					
						
							|  |  |  | #define MOC_NODE_RES    8
 | 
					
						
							|  |  |  | #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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 08:29:20 +00:00
										 |  |  | static int mesh_octree_get_base_offs(const float co[3], const float offs[3], const float div[3]) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int vx, vy, vz; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	vx = floor( (co[0] - offs[0]) / div[0]); | 
					
						
							|  |  |  | 	vy = floor( (co[1] - offs[1]) / div[1]); | 
					
						
							|  |  |  | 	vz = floor( (co[2] - offs[2]) / div[2]); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	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; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mesh_octree_add_node(MocNode **bt, intptr_t index) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (*bt == NULL) { | 
					
						
							|  |  |  | 		*bt = MEM_callocN(sizeof(MocNode), "MocNode"); | 
					
						
							|  |  |  | 		(*bt)->index[0] = index; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		int a; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		for (a = 0; a < MOC_NODE_RES; a++) { | 
					
						
							|  |  |  | 			if ((*bt)->index[a] == index) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				return; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			else if ((*bt)->index[a] == 0) { | 
					
						
							|  |  |  | 				(*bt)->index[a] = index; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		mesh_octree_add_node(&(*bt)->next, index); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mesh_octree_free_node(MocNode **bt) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if ( (*bt)->next) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		mesh_octree_free_node(&(*bt)->next); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	MEM_freeN(*bt); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* temporal define, just to make nicer code below */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | #define MOC_INDEX(vx, vy, vz)  (((vx) * MOC_RES * MOC_RES) + (vy) * MOC_RES + (vz))
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (!finite(co[0]) || | 
					
						
							|  |  |  | 	    !finite(co[1]) || | 
					
						
							|  |  |  | 	    !finite(co[2])) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	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 = (int)floorf(fx); | 
					
						
							|  |  |  | 	vy = (int)floorf(fy); | 
					
						
							|  |  |  | 	vz = (int)floorf(fz); | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	mesh_octree_add_node(basetable + MOC_INDEX(vx, vy, vz), index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (vx > 0) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (fx - ((float)vx) - MOC_THRESH < 0.0f) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 			mesh_octree_add_node(basetable + MOC_INDEX(vx - 1, vy, vz), index); | 
					
						
							|  |  |  | 	if (vx < MOC_RES - 2) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (fx - ((float)vx) + MOC_THRESH > 1.0f) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 			mesh_octree_add_node(basetable + MOC_INDEX(vx + 1, vy, vz), index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (vy > 0) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (fy - ((float)vy) - MOC_THRESH < 0.0f) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 			mesh_octree_add_node(basetable + MOC_INDEX(vx, vy - 1, vz), index); | 
					
						
							|  |  |  | 	if (vy < MOC_RES - 2) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (fy - ((float)vy) + MOC_THRESH > 1.0f) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 			mesh_octree_add_node(basetable + MOC_INDEX(vx, vy + 1, vz), index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (vz > 0) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (fz - ((float)vz) - MOC_THRESH < 0.0f) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 			mesh_octree_add_node(basetable + MOC_INDEX(vx, vy, vz - 1), index); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (vz < MOC_RES - 2) | 
					
						
							|  |  |  | 		if (fz - ((float)vz) + MOC_THRESH > 1.0f) | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 			mesh_octree_add_node(basetable + MOC_INDEX(vx, vy, vz + 1), index); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 08:29:20 +00:00
										 |  |  | static intptr_t mesh_octree_find_index(MocNode **bt, MVert *mvert, const float co[3]) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float *vec; | 
					
						
							|  |  |  | 	int a; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (*bt == NULL) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	for (a = 0; a < MOC_NODE_RES; a++) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if ((*bt)->index[a]) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			/* does mesh verts and editmode, code looks potential dangerous, octree should really be filled OK! */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (mvert) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				vec = (mvert + (*bt)->index[a] - 1)->co; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 				if (compare_v3v3(vec, co, MOC_THRESH)) | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					return (*bt)->index[a] - 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 				BMVert *eve = (BMVert *)((*bt)->index[a]); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +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; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if ( (*bt)->next) | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +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-11-01 00:06:53 +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 */ | 
					
						
							| 
									
										
										
										
											2012-04-28 08:29:20 +00:00
										 |  |  | intptr_t mesh_octree_table(Object *ob, BMEditMesh *em, const float co[3], char mode) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	MocNode **bt; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (mode == 'u') {        /* use table */ | 
					
						
							|  |  |  | 		if (MeshOctree.table == NULL) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			mesh_octree_table(ob, em, NULL, 's'); | 
					
						
							| 
									
										
										
										
											2011-12-20 10:05:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (MeshOctree.table) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			Mesh *me = ob->data; | 
					
						
							|  |  |  | 			bt = MeshOctree.table + mesh_octree_get_base_offs(co, MeshOctree.offs, MeshOctree.div); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			if (em) | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 				return mesh_octree_find_index(bt, NULL, co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			else | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 				return mesh_octree_find_index(bt, me->mvert, co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	else if (mode == 's') {   /* start table */ | 
					
						
							|  |  |  | 		Mesh *me = ob->data; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (em && me->edit_btmesh == em) { | 
					
						
							| 
									
										
										
										
											2009-05-16 16:18:08 +00:00
										 |  |  | 			BMIter iter; | 
					
						
							|  |  |  | 			BMVert *eve; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_MESH (eve, &iter, em->bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-03-22 01:35:13 +00:00
										 |  |  | 				DO_MINMAX(eve->co, min, max); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else {		 | 
					
						
							|  |  |  | 			MVert *mvert; | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 			int a; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			for (a = 0, mvert = me->mvert; a < me->totvert; a++, mvert++) | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 				DO_MINMAX(mvert->co, min, max); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* for quick unit coordinate calculus */ | 
					
						
							| 
									
										
										
										
											2012-03-09 06:04:17 +00:00
										 |  |  | 		copy_v3_v3(MeshOctree.offs, min); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		MeshOctree.offs[0] -= MOC_THRESH;        /* we offset it 1 threshold unit extra */ | 
					
						
							|  |  |  | 		MeshOctree.offs[1] -= MOC_THRESH; | 
					
						
							|  |  |  | 		MeshOctree.offs[2] -= MOC_THRESH; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		sub_v3_v3v3(MeshOctree.div, max, min); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		mul_v3_fl(MeshOctree.div, 1.0f / MOC_RES); | 
					
						
							|  |  |  | 		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; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (MeshOctree.table) /* happens when entering this call without ending it */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			mesh_octree_table(ob, em, co, 'e'); | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		MeshOctree.table = MEM_callocN(MOC_RES * MOC_RES * MOC_RES * sizeof(void *), "sym table"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (em && me->edit_btmesh == em) { | 
					
						
							| 
									
										
										
										
											2009-05-16 16:18:08 +00:00
										 |  |  | 			BMVert *eve; | 
					
						
							|  |  |  | 			BMIter iter; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_MESH (eve, &iter, em->bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				mesh_octree_add_nodes(MeshOctree.table, eve->co, MeshOctree.offs, MeshOctree.div, (intptr_t)(eve)); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else {		 | 
					
						
							|  |  |  | 			MVert *mvert; | 
					
						
							|  |  |  | 			int a; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	else if (mode == 'e') { /* end table */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (MeshOctree.table) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			int a; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			for (a = 0, bt = MeshOctree.table; a < MOC_RES * MOC_RES * MOC_RES; a++, bt++) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 				if (*bt) mesh_octree_free_node(bt); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			MEM_freeN(MeshOctree.table); | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			MeshOctree.table = NULL; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | MirrTopoStore_t mesh_topo_store = {NULL, -1. - 1, -1}; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* mode is 's' start, or 'e' end, or 'u' use */ | 
					
						
							|  |  |  | /* if end, ob can be NULL */ | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | /* note, is supposed return -1 on error, which callers are currently checking for, but is not used so far */ | 
					
						
							|  |  |  | int mesh_mirrtopo_table(Object *ob, char mode) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (mode == 'u') {        /* use table */ | 
					
						
							| 
									
										
										
										
											2011-12-20 10:05:58 +00:00
										 |  |  | 		if (ED_mesh_mirrtopo_recalc_check(ob->data, ob->mode, &mesh_topo_store)) { | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 			mesh_mirrtopo_table(ob, 's'); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-12-20 10:05:58 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	else if (mode == 's') { /* start table */ | 
					
						
							| 
									
										
										
										
											2011-12-20 22:01:11 +00:00
										 |  |  | 		ED_mesh_mirrtopo_init(ob->data, ob->mode, &mesh_topo_store, FALSE); | 
					
						
							| 
									
										
										
										
											2011-12-20 10:05:58 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	else if (mode == 'e') { /* end table */ | 
					
						
							| 
									
										
										
										
											2011-12-20 10:05:58 +00:00
										 |  |  | 		ED_mesh_mirrtopo_free(&mesh_topo_store); | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-14 17:55:27 +00:00
										 |  |  | static int mesh_get_x_mirror_vert_spacial(Object *ob, int index) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	Mesh *me = ob->data; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MVert *mvert; | 
					
						
							|  |  |  | 	float vec[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (mesh_mirrtopo_table(ob, 'u') == -1) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-20 10:05:58 +00:00
										 |  |  | 	return mesh_topo_store.index_lookup[index]; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		return mesh_get_x_mirror_vert_spacial(ob, index); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-03-09 04:32:40 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 08:29:20 +00:00
										 |  |  | static BMVert *editbmesh_get_x_mirror_vert_spacial(Object *ob, BMEditMesh *em, const float co[3]) | 
					
						
							| 
									
										
										
										
											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]) || | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	    !finite(co[1]) || | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	    !finite(co[2])) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	vec[0] = -co[0]; | 
					
						
							|  |  |  | 	vec[1] = co[1]; | 
					
						
							|  |  |  | 	vec[2] = co[2]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	poinval = mesh_octree_table(ob, em, vec, 'u'); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (poinval != -1) | 
					
						
							| 
									
										
										
										
											2009-05-18 10:29:37 +00:00
										 |  |  | 		return (BMVert *)(poinval); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | static BMVert *editbmesh_get_x_mirror_vert_topo(Object *ob, struct BMEditMesh *em, BMVert *eve, int index) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-16 07:27:56 +00:00
										 |  |  | 	intptr_t poinval; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	if (mesh_mirrtopo_table(ob, 'u') == -1) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 	if (index == -1) { | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 		BMIter iter; | 
					
						
							|  |  |  | 		BMVert *v; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		index = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 		BM_ITER_MESH (v, &iter, em->bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 			if (v == eve) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			index++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (index == em->bm->totvert) { | 
					
						
							| 
									
										
										
										
											2010-09-28 19:53:45 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	poinval = mesh_topo_store.index_lookup[index]; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (poinval != -1) | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 		return (BMVert *)(poinval); | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2010-10-16 02:40:31 +00:00
										 |  |  | }	 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 08:29:20 +00:00
										 |  |  | BMVert *editbmesh_get_x_mirror_vert(Object *ob, struct BMEditMesh *em, BMVert *eve, const float co[3], int index) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (((Mesh *)ob->data)->editflag & ME_EDIT_MIRROR_TOPO) { | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 		return editbmesh_get_x_mirror_vert_topo(ob, em, eve, index); | 
					
						
							| 
									
										
										
										
											2012-02-06 09:39:47 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 		return editbmesh_get_x_mirror_vert_spacial(ob, em, co); | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-06 09:39:47 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-11 02:14:43 +00:00
										 |  |  | static float *editmesh_get_mirror_uv(BMEditMesh *em, int axis, float *uv, float *mirrCent, float *face_cent) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec[2]; | 
					
						
							|  |  |  | 	float cent_vec[2]; | 
					
						
							|  |  |  | 	float cent[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* ignore nan verts */ | 
					
						
							|  |  |  | 	if (isnan(uv[0]) || !finite(uv[0]) || | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	    isnan(uv[1]) || !finite(uv[1]) | 
					
						
							|  |  |  | 	    ) | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (axis) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		vec[0] = uv[0]; | 
					
						
							|  |  |  | 		vec[1] = -((uv[1]) - mirrCent[1]) + mirrCent[1]; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		cent_vec[0] = face_cent[0]; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		cent_vec[1] = -((face_cent[1]) - mirrCent[1]) + mirrCent[1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		vec[0] = -((uv[0]) - mirrCent[0]) + mirrCent[0]; | 
					
						
							|  |  |  | 		vec[1] = uv[1]; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		cent_vec[0] = -((face_cent[0]) - mirrCent[0]) + mirrCent[0]; | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 		cent_vec[1] = face_cent[1]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* TODO - Optimize */ | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 		BMIter iter; | 
					
						
							|  |  |  | 		BMFace *efa; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 		BM_ITER_MESH (efa, &iter, em->bm, BM_FACES_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-04-27 07:26:28 +00:00
										 |  |  | 			uv_poly_center(em, efa, cent); | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 			if ( (fabs(cent[0] - cent_vec[0]) < 0.001) && (fabs(cent[1] - cent_vec[1]) < 0.001) ) { | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 				BMIter liter; | 
					
						
							|  |  |  | 				BMLoop *l; | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (l, &liter, efa, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2011-02-27 06:19:40 +00:00
										 |  |  | 					MLoopUV *luv = CustomData_bmesh_get(&em->bm->ldata, l->head.data, CD_MLOOPUV); | 
					
						
							|  |  |  | 					if ( (fabs(luv->uv[0] - vec[0]) < 0.001) && (fabs(luv->uv[1] - vec[1]) < 0.001) ) { | 
					
						
							|  |  |  | 						return luv->uv; | 
					
						
							|  |  |  | 								 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2010-02-17 19:50:42 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-06 09:39:47 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static unsigned int mirror_facehash(const void *ptr) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	const MFace *mf = ptr; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	int v0, v1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (mf->v4) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		v0 = MIN4(mf->v1, mf->v2, mf->v3, mf->v4); | 
					
						
							|  |  |  | 		v1 = MAX4(mf->v1, mf->v2, mf->v3, mf->v4); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		v0 = MIN3(mf->v1, mf->v2, mf->v3); | 
					
						
							|  |  |  | 		v1 = MAX3(mf->v1, mf->v2, mf->v3); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	return ((v0 * 39) ^ (v1 * 31)); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int mirror_facerotation(MFace *a, MFace *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	if (b->v4) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (a->v1 == b->v1 && a->v2 == b->v2 && a->v3 == b->v3 && a->v4 == b->v4) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		else if (a->v4 == b->v1 && a->v1 == b->v2 && a->v2 == b->v3 && a->v3 == b->v4) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 1; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		else if (a->v3 == b->v1 && a->v4 == b->v2 && a->v1 == b->v3 && a->v2 == b->v4) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 2; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		else if (a->v2 == b->v1 && a->v3 == b->v2 && a->v4 == b->v3 && a->v1 == b->v4) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 3; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (a->v1 == b->v1 && a->v2 == b->v2 && a->v3 == b->v3) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		else if (a->v3 == b->v1 && a->v1 == b->v2 && a->v2 == b->v3) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 1; | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		else if (a->v2 == b->v1 && a->v3 == b->v2 && a->v1 == b->v3) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 17:05:21 +00:00
										 |  |  | static int mirror_facecmp(const void *a, const void *b) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	return (mirror_facerotation((MFace *)a, (MFace *)b) == -1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-15 14:39:18 +00:00
										 |  |  | /* BMESH_TODO, convert to MPoly (functions above also) */ | 
					
						
							| 
									
										
										
										
											2009-05-18 08:46:04 +00:00
										 |  |  | int *mesh_get_x_mirror_faces(Object *ob, BMEditMesh *em) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	Mesh *me = ob->data; | 
					
						
							|  |  |  | 	MVert *mv, *mvert = me->mvert; | 
					
						
							|  |  |  | 	MFace mirrormf, *mf, *hashmf, *mface = me->mface; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	GHash *fhash; | 
					
						
							|  |  |  | 	int *mirrorverts, *mirrorfaces; | 
					
						
							|  |  |  | 	int a; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	mirrorverts = MEM_callocN(sizeof(int) * me->totvert, "MirrorVerts"); | 
					
						
							|  |  |  | 	mirrorfaces = MEM_callocN(sizeof(int) * 2 * me->totface, "MirrorFaces"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	mesh_octree_table(ob, em, NULL, 's'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	for (a = 0, mv = mvert; a < me->totvert; a++, mv++) | 
					
						
							|  |  |  | 		mirrorverts[a] = mesh_get_x_mirror_vert(ob, a); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	mesh_octree_table(ob, em, NULL, 'e'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	fhash = BLI_ghash_new(mirror_facehash, mirror_facecmp, "mirror_facehash gh"); | 
					
						
							|  |  |  | 	for (a = 0, mf = mface; a < me->totface; a++, mf++) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		BLI_ghash_insert(fhash, mf, mf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 	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; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* make sure v4 is not 0 if a quad */ | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		if (mf->v4 && mirrormf.v4 == 0) { | 
					
						
							| 
									
										
										
										
											2011-12-27 03:54:23 +00:00
										 |  |  | 			SWAP(unsigned int, mirrormf.v1, mirrormf.v3); | 
					
						
							|  |  |  | 			SWAP(unsigned int, mirrormf.v2, mirrormf.v4); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 		hashmf = BLI_ghash_lookup(fhash, &mirrormf); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		if (hashmf) { | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			mirrorfaces[a * 2] = hashmf - mface; | 
					
						
							|  |  |  | 			mirrorfaces[a * 2 + 1] = mirror_facerotation(&mirrormf, hashmf); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2012-03-26 02:56:48 +00:00
										 |  |  | 			mirrorfaces[a * 2] = -1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_ghash_free(fhash, NULL, NULL); | 
					
						
							|  |  |  | 	MEM_freeN(mirrorverts); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return mirrorfaces; | 
					
						
							|  |  |  | } |