| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2010-08-31 22:08:01 +00:00
										 |  |  | * $Id$ | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +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, | 
					
						
							|  |  |  | * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | * The Original Code is Copyright (C) 2011 by Blender Foundation | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | * All rights reserved. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | * The Original Code is: all of this file. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | * Contributor(s): Benoit Bolsee, | 
					
						
							|  |  |  | *                 Nick Samarin | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | * | 
					
						
							|  |  |  | * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2011-02-16 17:07:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include <math.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_scene_types.h"
 | 
					
						
							|  |  |  | #include "DNA_object_types.h"
 | 
					
						
							|  |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-21 19:44:59 +00:00
										 |  |  | #include "DNA_modifier_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include "DNA_ID.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 13:20:13 +00:00
										 |  |  | #include "BKE_library.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include "BKE_depsgraph.h"
 | 
					
						
							|  |  |  | #include "BKE_context.h"
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | #include "BKE_main.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | #include "BKE_modifier.h"
 | 
					
						
							|  |  |  | #include "BKE_scene.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include "BKE_DerivedMesh.h"
 | 
					
						
							|  |  |  | #include "BKE_cdderivedmesh.h"
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include "BLI_editVert.h"
 | 
					
						
							|  |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2011-02-16 17:07:18 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | #include "BLI_math_vector.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | #include "ED_object.h"
 | 
					
						
							|  |  |  | #include "ED_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | #include "ED_screen.h"
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | #include "RNA_access.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | #include "WM_api.h"
 | 
					
						
							|  |  |  | #include "WM_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | #include "mesh_intern.h"
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | #include "recast-capi.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | static void createVertsTrisData(bContext *C, LinkNode* obs, int *nverts_r, float **verts_r, int *ntris_r, int **tris_r) | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	MVert *mvert; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	int nfaces= 0, *tri, i, curnverts, basenverts, curnfaces; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	MFace *mface; | 
					
						
							|  |  |  | 	float co[3], wco[3]; | 
					
						
							|  |  |  | 	Object *ob; | 
					
						
							|  |  |  | 	LinkNode *oblink, *dmlink; | 
					
						
							|  |  |  | 	DerivedMesh *dm; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	Scene* scene= CTX_data_scene(C); | 
					
						
							|  |  |  | 	LinkNode* dms= NULL; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	int nverts, ntris, *tris; | 
					
						
							|  |  |  | 	float *verts; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	nverts= 0; | 
					
						
							|  |  |  | 	ntris= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* calculate number of verts and tris */ | 
					
						
							|  |  |  | 	for(oblink= obs; oblink; oblink= oblink->next) { | 
					
						
							|  |  |  | 		ob= (Object*) oblink->link; | 
					
						
							|  |  |  | 		dm= mesh_create_derived_no_virtual(scene, ob, NULL, CD_MASK_MESH); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 		BLI_linklist_append(&dms, (void*)dm); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 		nverts+= dm->getNumVerts(dm); | 
					
						
							|  |  |  | 		nfaces= dm->getNumFaces(dm); | 
					
						
							|  |  |  | 		ntris+= nfaces; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 		/* resolve quad faces */ | 
					
						
							|  |  |  | 		mface= dm->getFaceArray(dm); | 
					
						
							|  |  |  | 		for(i= 0; i<nfaces; i++) { | 
					
						
							|  |  |  | 			MFace* mf= &mface[i]; | 
					
						
							|  |  |  | 			if(mf->v4) | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 				ntris+=1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* create data */ | 
					
						
							|  |  |  | 	verts= MEM_mallocN(sizeof(float)*3*nverts, "createVertsTrisData verts"); | 
					
						
							|  |  |  | 	tris= MEM_mallocN(sizeof(int)*3*ntris, "createVertsTrisData faces"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	basenverts= 0; | 
					
						
							|  |  |  | 	tri= tris; | 
					
						
							|  |  |  | 	for(oblink= obs, dmlink= dms; oblink && dmlink; | 
					
						
							|  |  |  | 			oblink= oblink->next, dmlink= dmlink->next) { | 
					
						
							|  |  |  | 		ob= (Object*) oblink->link; | 
					
						
							|  |  |  | 		dm= (DerivedMesh*) dmlink->link; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		curnverts= dm->getNumVerts(dm); | 
					
						
							|  |  |  | 		mvert= dm->getVertArray(dm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* copy verts */ | 
					
						
							|  |  |  | 		for(i= 0; i<curnverts; i++) { | 
					
						
							|  |  |  | 			MVert *v= &mvert[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 			copy_v3_v3(co, v->co); | 
					
						
							|  |  |  | 			mul_v3_m4v3(wco, ob->obmat, co); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			verts[3*(basenverts+i)+0]= wco[0]; | 
					
						
							|  |  |  | 			verts[3*(basenverts+i)+1]= wco[2]; | 
					
						
							|  |  |  | 			verts[3*(basenverts+i)+2]= wco[1]; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 		/* create tris */ | 
					
						
							|  |  |  | 		curnfaces= dm->getNumFaces(dm); | 
					
						
							|  |  |  | 		mface= dm->getFaceArray(dm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for(i= 0; i<curnfaces; i++) { | 
					
						
							|  |  |  | 			MFace* mf= &mface[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			tri[0]= basenverts + mf->v1; | 
					
						
							|  |  |  | 			tri[1]= basenverts + mf->v3; | 
					
						
							|  |  |  | 			tri[2]= basenverts + mf->v2; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 			tri += 3; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if(mf->v4) { | 
					
						
							|  |  |  | 				tri[0]= basenverts + mf->v1; | 
					
						
							|  |  |  | 				tri[1]= basenverts + mf->v4; | 
					
						
							|  |  |  | 				tri[2]= basenverts + mf->v3; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 				tri += 3; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		basenverts+= curnverts; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* release derived mesh */ | 
					
						
							|  |  |  | 	for(dmlink= dms; dmlink; dmlink= dmlink->next) { | 
					
						
							|  |  |  | 		dm= (DerivedMesh*) dmlink->link; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 		dm->release(dm); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	BLI_linklist_free(dms, NULL); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	*nverts_r= nverts; | 
					
						
							|  |  |  | 	*verts_r= verts; | 
					
						
							|  |  |  | 	*ntris_r= ntris; | 
					
						
							|  |  |  | 	*tris_r= tris; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | static int buildNavMesh(const RecastData *recastParams, int nverts, float *verts, int ntris, int *tris, | 
					
						
							|  |  |  | 								 struct recast_polyMesh **pmesh, struct recast_polyMeshDetail **dmesh) | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float bmin[3], bmax[3]; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	struct recast_heightfield *solid; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	unsigned char *triflags; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	struct recast_compactHeightfield* chf; | 
					
						
							|  |  |  | 	struct recast_contourSet *cset; | 
					
						
							|  |  |  | 	int width, height, walkableHeight, walkableClimb, walkableRadius; | 
					
						
							|  |  |  | 	int minRegionSize, mergeRegionSize, maxEdgeLen; | 
					
						
							|  |  |  | 	float detailSampleDist, detailSampleMaxError; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_calcBounds(verts, nverts, bmin, bmax); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 1. Initialize build config ** */ | 
					
						
							|  |  |  | 	walkableHeight= (int)ceilf(recastParams->agentheight/ recastParams->cellheight); | 
					
						
							|  |  |  | 	walkableClimb= (int)floorf(recastParams->agentmaxclimb / recastParams->cellheight); | 
					
						
							|  |  |  | 	walkableRadius= (int)ceilf(recastParams->agentradius / recastParams->cellsize); | 
					
						
							|  |  |  | 	minRegionSize= (int)(recastParams->regionminsize * recastParams->regionminsize); | 
					
						
							|  |  |  | 	mergeRegionSize= (int)(recastParams->regionmergesize * recastParams->regionmergesize); | 
					
						
							|  |  |  | 	maxEdgeLen= (int)(recastParams->edgemaxlen/recastParams->cellsize); | 
					
						
							|  |  |  | 	detailSampleDist= recastParams->detailsampledist< 0.9f ? 0 : | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 			recastParams->cellsize * recastParams->detailsampledist; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	detailSampleMaxError= recastParams->cellheight * recastParams->detailsamplemaxerror; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* Set the area where the navigation will be build. */ | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_calcGridSize(bmin, bmax, recastParams->cellsize, &width, &height); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 2: Rasterize input polygon soup ** */ | 
					
						
							|  |  |  | 	/* Allocate voxel heightfield where we rasterize our input data to */ | 
					
						
							|  |  |  | 	solid= recast_newHeightfield(); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	if(!recast_createHeightfield(solid, width, height, bmin, bmax, recastParams->cellsize, recastParams->cellheight)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyHeightfield(solid); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* Allocate array that can hold triangle flags */ | 
					
						
							|  |  |  | 	triflags= MEM_callocN(sizeof(unsigned char)*ntris, "buildNavMesh triflags"); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* Find triangles which are walkable based on their slope and rasterize them */ | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_markWalkableTriangles(RAD2DEG(recastParams->agentmaxslope), verts, nverts, tris, ntris, triflags); | 
					
						
							|  |  |  | 	recast_rasterizeTriangles(verts, nverts, tris, triflags, ntris, solid); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	MEM_freeN(triflags); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 3: Filter walkables surfaces ** */ | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_filterLedgeSpans(walkableHeight, walkableClimb, solid); | 
					
						
							|  |  |  | 	recast_filterWalkableLowHeightSpans(walkableHeight, solid); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 4: Partition walkable surface to simple regions ** */ | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	chf= recast_newCompactHeightfield(); | 
					
						
							|  |  |  | 	if(!recast_buildCompactHeightfield(walkableHeight, walkableClimb, RECAST_WALKABLE, solid, chf)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyHeightfield(solid); | 
					
						
							|  |  |  | 		recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_destroyHeightfield(solid); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* Prepare for region partitioning, by calculating distance field along the walkable surface */ | 
					
						
							|  |  |  | 	if(!recast_buildDistanceField(chf)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* Partition the walkable surface into simple regions without holes */ | 
					
						
							|  |  |  | 	if(!recast_buildRegions(chf, walkableRadius, 0, minRegionSize, mergeRegionSize)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 5: Trace and simplify region contours ** */ | 
					
						
							|  |  |  | 	/* Create contours */ | 
					
						
							|  |  |  | 	cset= recast_newContourSet(); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	if(!recast_buildContours(chf, recastParams->edgemaxerror, maxEdgeLen, cset)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 		recast_destroyContourSet(cset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 6: Build polygons mesh from contours ** */ | 
					
						
							|  |  |  | 	*pmesh= recast_newPolyMesh(); | 
					
						
							|  |  |  | 	if(!recast_buildPolyMesh(cset, recastParams->vertsperpoly, *pmesh)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 		recast_destroyContourSet(cset); | 
					
						
							|  |  |  | 		recast_destroyPolyMesh(*pmesh); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* ** Step 7: Create detail mesh which allows to access approximate height on each polygon ** */ | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	*dmesh= recast_newPolyMeshDetail(); | 
					
						
							|  |  |  | 	if(!recast_buildPolyMeshDetail(*pmesh, chf, detailSampleDist, detailSampleMaxError, *dmesh)) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 		recast_destroyContourSet(cset); | 
					
						
							|  |  |  | 		recast_destroyPolyMesh(*pmesh); | 
					
						
							|  |  |  | 		recast_destroyPolyMeshDetail(*dmesh); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_destroyCompactHeightfield(chf); | 
					
						
							|  |  |  | 	recast_destroyContourSet(cset); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	return 1; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | static Object* createRepresentation(bContext *C, struct recast_polyMesh *pmesh, struct recast_polyMeshDetail *dmesh, Base* base) | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float co[3], rot[3]; | 
					
						
							|  |  |  | 	EditMesh *em; | 
					
						
							| 
									
										
										
										
											2011-09-10 09:38:38 +00:00
										 |  |  | 	int i,j, k; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	unsigned short* v; | 
					
						
							|  |  |  | 	int face[3]; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	Main *bmain= CTX_data_main(C); | 
					
						
							| 
									
										
										
										
											2010-07-21 19:44:59 +00:00
										 |  |  | 	Scene *scene= CTX_data_scene(C); | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 	Object* obedit; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	int createob= base==NULL; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	int nverts, nmeshes, nvp; | 
					
						
							|  |  |  | 	unsigned short *verts, *meshes, *polys; | 
					
						
							|  |  |  | 	float bmin[3], cs, ch, *dverts; | 
					
						
							|  |  |  | 	unsigned char *tris; | 
					
						
							|  |  |  | 	ModifierData *md; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	zero_v3(co); | 
					
						
							|  |  |  | 	zero_v3(rot); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	if(createob) { | 
					
						
							|  |  |  | 		/* create new object */ | 
					
						
							|  |  |  | 		obedit= ED_object_add_type(C, OB_MESH, co, rot, FALSE, 1); | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		obedit= base->object; | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 		scene_select_base(scene, base); | 
					
						
							|  |  |  | 		copy_v3_v3(obedit->loc, co); | 
					
						
							|  |  |  | 		copy_v3_v3(obedit->rot, rot); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 	ED_object_enter_editmode(C, EM_DO_UNDO|EM_IGNORE_LAYER); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	em= BKE_mesh_get_editmesh(((Mesh *)obedit->data)); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	if(!createob) { | 
					
						
							|  |  |  | 		/* clear */ | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 		if(em->verts.first) free_vertlist(em, &em->verts); | 
					
						
							|  |  |  | 		if(em->edges.first) free_edgelist(em, &em->edges); | 
					
						
							|  |  |  | 		if(em->faces.first) free_facelist(em, &em->faces); | 
					
						
							|  |  |  | 		if(em->selected.first) BLI_freelistN(&(em->selected)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* create verts for polygon mesh */ | 
					
						
							|  |  |  | 	verts= recast_polyMeshGetVerts(pmesh, &nverts); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_polyMeshGetBoundbox(pmesh, bmin, NULL); | 
					
						
							|  |  |  | 	recast_polyMeshGetCell(pmesh, &cs, &ch); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	for(i= 0; i<nverts; i++) { | 
					
						
							|  |  |  | 		v= &verts[3*i]; | 
					
						
							|  |  |  | 		co[0]= bmin[0] + v[0]*cs; | 
					
						
							|  |  |  | 		co[1]= bmin[1] + v[1]*ch; | 
					
						
							|  |  |  | 		co[2]= bmin[2] + v[2]*cs; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 		SWAP(float, co[1], co[2]); | 
					
						
							|  |  |  | 		addvertlist(em, co, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* create custom data layer to save polygon idx */ | 
					
						
							|  |  |  | 	CustomData_add_layer_named(&em->fdata, CD_RECAST, CD_CALLOC, NULL, 0, "createRepresentation recastData"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create verts and faces for detailed mesh */ | 
					
						
							|  |  |  | 	meshes= recast_polyMeshDetailGetMeshes(dmesh, &nmeshes); | 
					
						
							|  |  |  | 	polys= recast_polyMeshGetPolys(pmesh, NULL, &nvp); | 
					
						
							|  |  |  | 	dverts= recast_polyMeshDetailGetVerts(dmesh, NULL); | 
					
						
							|  |  |  | 	tris= recast_polyMeshDetailGetTris(dmesh, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i= 0; i<nmeshes; i++) { | 
					
						
							|  |  |  | 		int uniquevbase= em->totvert; | 
					
						
							|  |  |  | 		unsigned short vbase= meshes[4*i+0]; | 
					
						
							|  |  |  | 		unsigned short ndv= meshes[4*i+1]; | 
					
						
							|  |  |  | 		unsigned short tribase= meshes[4*i+2]; | 
					
						
							|  |  |  | 		unsigned short trinum= meshes[4*i+3]; | 
					
						
							|  |  |  | 		const unsigned short* p= &polys[i*nvp*2]; | 
					
						
							|  |  |  | 		int nv= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for(j= 0; j < nvp; ++j) { | 
					
						
							|  |  |  | 			if(p[j]==0xffff) break; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 			nv++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* create unique verts  */ | 
					
						
							|  |  |  | 		for(j= nv; j<ndv; j++) { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 			copy_v3_v3(co, &dverts[3*(vbase + j)]); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 			SWAP(float, co[1], co[2]); | 
					
						
							|  |  |  | 			addvertlist(em, co, NULL); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		EM_init_index_arrays(em, 1, 0, 0); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* create faces */ | 
					
						
							|  |  |  | 		for(j= 0; j<trinum; j++) { | 
					
						
							|  |  |  | 			unsigned char* tri= &tris[4*(tribase+j)]; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 			EditFace* newFace; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 			int* polygonIdx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 			for(k= 0; k<3; k++) { | 
					
						
							|  |  |  | 				if(tri[k]<nv) | 
					
						
							|  |  |  | 					face[k]= p[tri[k]]; /* shared vertex */ | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 				else | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 					face[k]= uniquevbase+tri[k]-nv; /* unique vertex */ | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 			newFace= addfacelist(em, EM_get_vert_for_index(face[0]), EM_get_vert_for_index(face[2]), | 
					
						
							| 
									
										
										
										
											2010-07-20 14:09:05 +00:00
										 |  |  | 									EM_get_vert_for_index(face[1]), NULL, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 			/* set navigation polygon idx to the custom layer */ | 
					
						
							|  |  |  | 			polygonIdx= (int*)CustomData_em_get(&em->fdata, newFace->data, CD_RECAST); | 
					
						
							|  |  |  | 			*polygonIdx= i+1; /* add 1 to avoid zero idx */ | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		EM_free_index_arrays(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	recast_destroyPolyMesh(pmesh); | 
					
						
							|  |  |  | 	recast_destroyPolyMeshDetail(dmesh); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh((Mesh*)obedit->data, em); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2011-02-16 17:07:18 +00:00
										 |  |  | 	DAG_id_tag_update((ID*)obedit->data, OB_RECALC_DATA); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ED_object_exit_editmode(C, EM_FREEDATA);  | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_OBJECT|ND_DRAW, obedit); | 
					
						
							| 
									
										
										
										
											2010-07-14 13:20:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	if(createob) { | 
					
						
							|  |  |  | 		obedit->gameflag&= ~OB_COLLISION; | 
					
						
							|  |  |  | 		obedit->gameflag|= OB_NAVMESH; | 
					
						
							|  |  |  | 		obedit->body_type= OB_BODY_TYPE_NAVMESH; | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 		rename_id((ID *)obedit, "Navmesh"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	md= modifiers_findByType(obedit, eModifierType_NavMesh); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	if(!md) { | 
					
						
							| 
									
										
										
										
											2010-08-05 13:54:56 +00:00
										 |  |  | 		ED_object_modifier_add(NULL, bmain, scene, obedit, NULL, eModifierType_NavMesh); | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-21 19:44:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	return obedit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | static int create_navmesh_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	Scene* scene= CTX_data_scene(C); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	int nverts, ntris; | 
					
						
							|  |  |  | 	float* verts; | 
					
						
							|  |  |  | 	int* tris; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	struct recast_polyMesh *pmesh; | 
					
						
							|  |  |  | 	struct recast_polyMeshDetail *dmesh; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	LinkNode* obs= NULL; | 
					
						
							|  |  |  | 	Base* navmeshBase= NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	CTX_DATA_BEGIN(C, Base*, base, selected_editable_bases) { | 
					
						
							|  |  |  | 		if(base->object->body_type==OB_BODY_TYPE_NAVMESH) { | 
					
						
							|  |  |  | 			if(!navmeshBase || base==CTX_data_active_base(C)) | 
					
						
							|  |  |  | 				navmeshBase= base; | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			BLI_linklist_append(&obs, (void*)base->object); | 
					
						
							| 
									
										
										
										
											2011-02-16 17:07:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	CTX_DATA_END; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	createVertsTrisData(C, obs, &nverts, &verts, &ntris, &tris); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	BLI_linklist_free(obs, NULL); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 	buildNavMesh(&scene->gm.recastData, nverts, verts, ntris, tris, &pmesh, &dmesh); | 
					
						
							| 
									
										
										
										
											2010-07-29 14:06:48 +00:00
										 |  |  | 	createRepresentation(C, pmesh, dmesh, navmeshBase); | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | 	MEM_freeN(verts); | 
					
						
							|  |  |  | 	MEM_freeN(tris); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | void MESH_OT_create_navmesh(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	ot->name= "Create navigation mesh"; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 	ot->description= "Create navigation mesh for selected objects"; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | 	ot->idname= "MESH_OT_create_navmesh"; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= create_navmesh_exec; | 
					
						
							| 
									
										
										
										
											2010-07-14 20:09:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2010-07-09 22:16:52 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | static int assign_navpolygon_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* do work here */ | 
					
						
							|  |  |  | 	int targetPolyIdx= -1; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	EditFace *ef, *efa; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	efa= EM_get_actFace(em, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(efa) { | 
					
						
							|  |  |  | 		if(CustomData_has_layer(&em->fdata, CD_RECAST)) { | 
					
						
							|  |  |  | 			targetPolyIdx= *(int*)CustomData_em_get(&em->fdata, efa->data, CD_RECAST); | 
					
						
							|  |  |  | 			targetPolyIdx= targetPolyIdx>=0? targetPolyIdx : -targetPolyIdx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(targetPolyIdx>0) { | 
					
						
							|  |  |  | 				/* set target poly idx to other selected faces */ | 
					
						
							|  |  |  | 				ef= (EditFace*)em->faces.last; | 
					
						
							|  |  |  | 				while(ef)  { | 
					
						
							|  |  |  | 					if((ef->f & SELECT )&& ef!=efa)  { | 
					
						
							|  |  |  | 						int* recastDataBlock= (int*)CustomData_em_get(&em->fdata, ef->data, CD_RECAST); | 
					
						
							|  |  |  | 						*recastDataBlock= targetPolyIdx; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 					ef= ef->prev; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}		 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2011-02-16 17:07:18 +00:00
										 |  |  | 	DAG_id_tag_update((ID*)obedit->data, OB_RECALC_DATA); | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh((Mesh*)obedit->data, em); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | void MESH_OT_assign_navpolygon(struct wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2011-09-10 09:21:46 +00:00
										 |  |  | 	ot->name= "Assign polygon index"; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	ot->description= "Assign polygon index to face by active face"; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | 	ot->idname= "MESH_OT_assign_navpolygon"; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	ot->exec= assign_navpolygon_exec; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int compare(const void * a, const void * b){   | 
					
						
							|  |  |  | 	return ( *(int*)a - *(int*)b ); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | static int findFreeNavPolyIndex(EditMesh* em) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* construct vector of indices */ | 
					
						
							|  |  |  | 	int numfaces= em->totface; | 
					
						
							|  |  |  | 	int* indices= MEM_callocN(sizeof(int)*numfaces, "findFreeNavPolyIndex(indices)"); | 
					
						
							|  |  |  | 	EditFace* ef= (EditFace*)em->faces.last; | 
					
						
							|  |  |  | 	int i, idx= 0, freeIdx= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(ef) { | 
					
						
							|  |  |  | 		int polyIdx= *(int*)CustomData_em_get(&em->fdata, ef->data, CD_RECAST); | 
					
						
							|  |  |  | 		indices[idx]= polyIdx; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 		idx++; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 		ef= ef->prev; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	qsort(indices, numfaces, sizeof(int), compare); | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	/* search first free index */ | 
					
						
							|  |  |  | 	freeIdx= 1; | 
					
						
							|  |  |  | 	for(i= 0; i<numfaces; i++) { | 
					
						
							|  |  |  | 		if(indices[i]==freeIdx) | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 			freeIdx++; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 		else if(indices[i]>freeIdx) | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(indices); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	return freeIdx; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:43 +00:00
										 |  |  | static int assign_new_navpolygon_exec(bContext *C, wmOperator *UNUSED(op)) | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							|  |  |  | 	EditFace *ef; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(CustomData_has_layer(&em->fdata, CD_RECAST)) { | 
					
						
							|  |  |  | 		int targetPolyIdx= findFreeNavPolyIndex(em); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(targetPolyIdx>0) { | 
					
						
							|  |  |  | 			/* set target poly idx to selected faces */ | 
					
						
							|  |  |  | 			ef= (EditFace*)em->faces.last; | 
					
						
							|  |  |  | 			while(ef) { | 
					
						
							|  |  |  | 				if(ef->f & SELECT) { | 
					
						
							|  |  |  | 					int *recastDataBlock= (int*)CustomData_em_get(&em->fdata, ef->data, CD_RECAST); | 
					
						
							|  |  |  | 					*recastDataBlock= targetPolyIdx; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 				ef= ef->prev; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:55 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 17:07:18 +00:00
										 |  |  | 	DAG_id_tag_update((ID*)obedit->data, OB_RECALC_DATA); | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh((Mesh*)obedit->data, em); | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | void MESH_OT_assign_new_navpolygon(struct wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2011-09-10 09:21:46 +00:00
										 |  |  | 	ot->name= "Assign new polygon index"; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	ot->description= "Assign new polygon index to face"; | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | 	ot->idname= "MESH_OT_assign_new_navpolygon"; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2011-09-27 09:09:52 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2010-07-27 21:01:00 +00:00
										 |  |  | 	ot->exec= assign_new_navpolygon_exec; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | } |