| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * $Id$ | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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 | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): none yet. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * (uit traces) maart 95 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-13 17:46:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BLI_editVert.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-13 17:46:30 +00:00
										 |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* callbacks for errors and interrupts and some goo */ | 
					
						
							|  |  |  | static void (*BLI_localErrorCallBack)(char*) = NULL; | 
					
						
							|  |  |  | static int (*BLI_localInterruptCallBack)(void) = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_setErrorCallBack(void (*f)(char*)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_localErrorCallBack = f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_setInterruptCallBack(int (*f)(void)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_localInterruptCallBack = f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* just flush the error to /dev/null if the error handler is missing */ | 
					
						
							|  |  |  | void callLocalErrorCallBack(char* msg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (BLI_localErrorCallBack) { | 
					
						
							|  |  |  | 		BLI_localErrorCallBack(msg); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-18 16:21:32 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | /* ignore if the interrupt wasn't set */ | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static int callLocalInterruptCallBack(void) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (BLI_localInterruptCallBack) { | 
					
						
							|  |  |  | 		return BLI_localInterruptCallBack(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2006-02-18 16:21:32 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* local types */ | 
					
						
							|  |  |  | typedef struct PolyFill { | 
					
						
							|  |  |  | 	int edges,verts; | 
					
						
							|  |  |  | 	float min[3],max[3]; | 
					
						
							|  |  |  | 	short f,nr; | 
					
						
							|  |  |  | } PolyFill; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct ScFillVert { | 
					
						
							|  |  |  | 	EditVert *v1; | 
					
						
							|  |  |  | 	EditEdge *first,*last; | 
					
						
							|  |  |  | 	short f,f1; | 
					
						
							|  |  |  | } ScFillVert; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* local funcs */ | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-30 12:38:38 +00:00
										 |  |  | #define COMPLIMIT	0.00003
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static ScFillVert *scdata; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | ListBase fillvertbase = {0,0}; | 
					
						
							|  |  |  | ListBase filledgebase = {0,0}; | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | ListBase fillfacebase = {0,0}; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static short cox, coy; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | /* ****  FUBCTIONS FOR QSORT *************************** */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static int vergscdata(const void *a1, const void *a2) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	const ScFillVert *x1=a1,*x2=a2; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if( x1->v1->co[coy] < x2->v1->co[coy] ) return 1; | 
					
						
							|  |  |  | 	else if( x1->v1->co[coy] > x2->v1->co[coy]) return -1; | 
					
						
							|  |  |  | 	else if( x1->v1->co[cox] > x2->v1->co[cox] ) return 1; | 
					
						
							|  |  |  | 	else if( x1->v1->co[cox] < x2->v1->co[cox]) return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static int vergpoly(const void *a1, const void *a2) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	const PolyFill *x1=a1, *x2=a2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if( x1->min[cox] > x2->min[cox] ) return 1; | 
					
						
							|  |  |  | 	else if( x1->min[cox] < x2->min[cox] ) return -1; | 
					
						
							|  |  |  | 	else if( x1->min[coy] > x2->min[coy] ) return 1; | 
					
						
							|  |  |  | 	else if( x1->min[coy] < x2->min[coy] ) return -1; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************* MEMORY MANAGEMENT ************* */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct mem_elements { | 
					
						
							|  |  |  | 	struct mem_elements *next, *prev; | 
					
						
							|  |  |  | 	char *data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* simple optimization for allocating thousands of small memory blocks
 | 
					
						
							|  |  |  |    only to be used within loops, and not by one function at a time | 
					
						
							|  |  |  |    free in the end, with argument '-1' | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static void *new_mem_element(int size) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int blocksize= 16384; | 
					
						
							| 
									
										
										
										
											2010-07-20 10:41:08 +00:00
										 |  |  | 	static int offs= 0;		/* the current free address */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	static struct mem_elements *cur= 0; | 
					
						
							|  |  |  | 	static ListBase lb= {0, 0}; | 
					
						
							|  |  |  | 	void *adr; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(size>10000 || size==0) { | 
					
						
							|  |  |  | 		printf("incorrect use of new_mem_element\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(size== -1) { | 
					
						
							|  |  |  | 		cur= lb.first; | 
					
						
							|  |  |  | 		while(cur) { | 
					
						
							|  |  |  | 			MEM_freeN(cur->data); | 
					
						
							|  |  |  | 			cur= cur->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		BLI_freelistN(&lb); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		return NULL;	 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	size= 4*( (size+3)/4 ); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(cur) { | 
					
						
							|  |  |  | 		if(size+offs < blocksize) { | 
					
						
							|  |  |  | 			adr= (void *) (cur->data+offs); | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 			 offs+= size; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			return adr; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	cur= MEM_callocN( sizeof(struct mem_elements), "newmem"); | 
					
						
							|  |  |  | 	cur->data= MEM_callocN(blocksize, "newmem"); | 
					
						
							|  |  |  | 	BLI_addtail(&lb, cur); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	offs= size; | 
					
						
							|  |  |  | 	return cur->data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_end_edgefill(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	new_mem_element(-1); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	fillvertbase.first= fillvertbase.last= 0; | 
					
						
							|  |  |  | 	filledgebase.first= filledgebase.last= 0; | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 	fillfacebase.first= fillfacebase.last= 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ****  FILL ROUTINES *************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | EditVert *BLI_addfillvert(float *vec) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	eve= new_mem_element(sizeof(EditVert)); | 
					
						
							|  |  |  | 	BLI_addtail(&fillvertbase, eve); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | 	eve->co[0] = vec[0]; | 
					
						
							|  |  |  | 	eve->co[1] = vec[1]; | 
					
						
							|  |  |  | 	eve->co[2] = vec[2]; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return eve;	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | EditEdge *BLI_addfilledge(EditVert *v1, EditVert *v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *newed; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	newed= new_mem_element(sizeof(EditEdge)); | 
					
						
							|  |  |  | 	BLI_addtail(&filledgebase, newed); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	newed->v1= v1; | 
					
						
							|  |  |  | 	newed->v2= v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return newed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static void addfillface(EditVert *v1, EditVert *v2, EditVert *v3, int mat_nr) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* does not make edges */ | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 	EditFace *evl; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 	evl= new_mem_element(sizeof(EditFace)); | 
					
						
							|  |  |  | 	BLI_addtail(&fillfacebase, evl); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	evl->v1= v1; | 
					
						
							|  |  |  | 	evl->v2= v2; | 
					
						
							|  |  |  | 	evl->v3= v3; | 
					
						
							|  |  |  | 	evl->f= 2; | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | 	evl->mat_nr= mat_nr; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static int boundisect(PolyFill *pf2, PolyFill *pf1) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* has pf2 been touched (intersected) by pf1 ? with bounding box */ | 
					
						
							|  |  |  | 	/* test first if polys exist */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(pf1->edges==0 || pf2->edges==0) return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(pf2->max[cox] < pf1->min[cox] ) return 0; | 
					
						
							|  |  |  | 	if(pf2->max[coy] < pf1->min[coy] ) return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(pf2->min[cox] > pf1->max[cox] ) return 0; | 
					
						
							|  |  |  | 	if(pf2->min[coy] > pf1->max[coy] ) return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* join */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	if(pf2->max[cox]<pf1->max[cox]) pf2->max[cox]= pf1->max[cox]; | 
					
						
							|  |  |  | 	if(pf2->max[coy]<pf1->max[coy]) pf2->max[coy]= pf1->max[coy]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(pf2->min[cox]>pf1->min[cox]) pf2->min[cox]= pf1->min[cox]; | 
					
						
							|  |  |  | 	if(pf2->min[coy]>pf1->min[coy]) pf2->min[coy]= pf1->min[coy]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static void mergepolysSimp(PolyFill *pf1, PolyFill *pf2)	/* add pf2 to pf1 */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* replace old poly numbers */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->xs== pf2->nr) eve->xs= pf1->nr; | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		if(eed->f1== pf2->nr) eed->f1= pf1->nr; | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pf1->verts+= pf2->verts; | 
					
						
							|  |  |  | 	pf1->edges+= pf2->edges; | 
					
						
							|  |  |  | 	pf2->verts= pf2->edges= 0; | 
					
						
							|  |  |  | 	pf1->f= (pf1->f | pf2->f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static short testedgeside(float *v1, float *v2, float *v3) | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | /* is v3 to the right of v1-v2 ? With exception: v3==v1 || v3==v2 */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float inp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	inp= (v2[cox]-v1[cox])*(v1[coy]-v3[coy]) | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 		+(v1[coy]-v2[coy])*(v1[cox]-v3[cox]); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(inp<0.0) return 0; | 
					
						
							|  |  |  | 	else if(inp==0) { | 
					
						
							|  |  |  | 		if(v1[cox]==v3[cox] && v1[coy]==v3[coy]) return 0; | 
					
						
							|  |  |  | 		if(v2[cox]==v3[cox] && v2[coy]==v3[coy]) return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static short addedgetoscanvert(ScFillVert *sc, EditEdge *eed) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* find first edge to the right of eed, and insert eed before that */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	EditEdge *ed; | 
					
						
							|  |  |  | 	float fac,fac1,x,y; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(sc->first==0) { | 
					
						
							|  |  |  | 		sc->first= sc->last= eed; | 
					
						
							|  |  |  | 		eed->prev= eed->next=0; | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	x= eed->v1->co[cox]; | 
					
						
							|  |  |  | 	y= eed->v1->co[coy]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fac1= eed->v2->co[coy]-y; | 
					
						
							|  |  |  | 	if(fac1==0.0) { | 
					
						
							|  |  |  | 		fac1= 1.0e10*(eed->v2->co[cox]-x); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else fac1= (x-eed->v2->co[cox])/fac1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ed= sc->first; | 
					
						
							|  |  |  | 	while(ed) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(ed->v2==eed->v2) return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		fac= ed->v2->co[coy]-y; | 
					
						
							|  |  |  | 		if(fac==0.0) { | 
					
						
							|  |  |  | 			fac= 1.0e10*(ed->v2->co[cox]-x); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else fac= (x-ed->v2->co[cox])/fac; | 
					
						
							|  |  |  | 		if(fac>fac1) break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ed= ed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(ed) BLI_insertlinkbefore((ListBase *)&(sc->first), ed, eed); | 
					
						
							|  |  |  | 	else BLI_addtail((ListBase *)&(sc->first),eed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static ScFillVert *addedgetoscanlist(EditEdge *eed, int len) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* inserts edge at correct location in ScFillVert list */ | 
					
						
							|  |  |  | 	/* returns sc when edge already exists */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	ScFillVert *sc,scsearch; | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* which vert is left-top? */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	if(eed->v1->co[coy] == eed->v2->co[coy]) { | 
					
						
							|  |  |  | 		if(eed->v1->co[cox] > eed->v2->co[cox]) { | 
					
						
							|  |  |  | 			eve= eed->v1; | 
					
						
							|  |  |  | 			eed->v1= eed->v2; | 
					
						
							|  |  |  | 			eed->v2= eve; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(eed->v1->co[coy] < eed->v2->co[coy]) { | 
					
						
							|  |  |  | 		eve= eed->v1; | 
					
						
							|  |  |  | 		eed->v1= eed->v2; | 
					
						
							|  |  |  | 		eed->v2= eve; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* find location in list */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	scsearch.v1= eed->v1; | 
					
						
							|  |  |  | 	sc= (ScFillVert *)bsearch(&scsearch,scdata,len, | 
					
						
							| 
									
										
										
										
											2010-03-22 09:30:00 +00:00
										 |  |  | 		sizeof(ScFillVert), vergscdata); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 01:52:28 +00:00
										 |  |  | 	if(sc==0) printf("Error in search edge: %p\n", (void *)eed); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	else if(addedgetoscanvert(sc,eed)==0) return sc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static short boundinsideEV(EditEdge *eed, EditVert *eve) | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | /* is eve inside boundbox eed */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float minx,maxx,miny,maxy; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(eed->v1->co[cox]<eed->v2->co[cox]) { | 
					
						
							|  |  |  | 		minx= eed->v1->co[cox]; | 
					
						
							|  |  |  | 		maxx= eed->v2->co[cox]; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		minx= eed->v2->co[cox]; | 
					
						
							|  |  |  | 		maxx= eed->v1->co[cox]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(eve->co[cox]>=minx && eve->co[cox]<=maxx) { | 
					
						
							|  |  |  | 		if(eed->v1->co[coy]<eed->v2->co[coy]) { | 
					
						
							|  |  |  | 			miny= eed->v1->co[coy]; | 
					
						
							|  |  |  | 			maxy= eed->v2->co[coy]; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			miny= eed->v2->co[coy]; | 
					
						
							|  |  |  | 			maxy= eed->v1->co[coy]; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(eve->co[coy]>=miny && eve->co[coy]<=maxy) return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static void testvertexnearedge(void) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* only vertices with ->h==1 are being tested for
 | 
					
						
							|  |  |  | 		being close to an edge, if true insert */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	EditEdge *eed,*ed1; | 
					
						
							|  |  |  | 	float dist,vec1[2],vec2[2],vec3[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->h==1) { | 
					
						
							|  |  |  | 			vec3[0]= eve->co[cox]; | 
					
						
							|  |  |  | 			vec3[1]= eve->co[coy]; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 			/* find the edge which has vertex eve */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			ed1= filledgebase.first; | 
					
						
							|  |  |  | 			while(ed1) { | 
					
						
							|  |  |  | 				if(ed1->v1==eve || ed1->v2==eve) break; | 
					
						
							|  |  |  | 				ed1= ed1->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(ed1->v1==eve) { | 
					
						
							|  |  |  | 				ed1->v1= ed1->v2; | 
					
						
							|  |  |  | 				ed1->v2= eve; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eed= filledgebase.first; | 
					
						
							|  |  |  | 			while(eed) { | 
					
						
							|  |  |  | 				if(eve!=eed->v1 && eve!=eed->v2 && eve->xs==eed->f1) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 					if(compare_v3v3(eve->co,eed->v1->co, COMPLIMIT)) { | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 						ed1->v2= eed->v1; | 
					
						
							|  |  |  | 						eed->v1->h++; | 
					
						
							|  |  |  | 						eve->h= 0; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 					else if(compare_v3v3(eve->co,eed->v2->co, COMPLIMIT)) { | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 						ed1->v2= eed->v2; | 
					
						
							|  |  |  | 						eed->v2->h++; | 
					
						
							|  |  |  | 						eve->h= 0; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						vec1[0]= eed->v1->co[cox]; | 
					
						
							|  |  |  | 						vec1[1]= eed->v1->co[coy]; | 
					
						
							|  |  |  | 						vec2[0]= eed->v2->co[cox]; | 
					
						
							|  |  |  | 						vec2[1]= eed->v2->co[coy]; | 
					
						
							|  |  |  | 						if(boundinsideEV(eed,eve)) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 							dist= dist_to_line_v2(vec1,vec2,vec3); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 							if(dist<COMPLIMIT) { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 								/* new edge */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 								ed1= BLI_addfilledge(eed->v1, eve); | 
					
						
							|  |  |  | 								 | 
					
						
							|  |  |  | 								/* printf("fill: vertex near edge %x\n",eve); */ | 
					
						
							|  |  |  | 								ed1->f= ed1->h= 0; | 
					
						
							|  |  |  | 								ed1->f1= eed->f1; | 
					
						
							|  |  |  | 								eed->v1= eve; | 
					
						
							|  |  |  | 								eve->h= 3; | 
					
						
							|  |  |  | 								break; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				eed= eed->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static void splitlist(ListBase *tempve, ListBase *temped, short nr) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* everything is in templist, write only poly nr to fillist */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	EditVert *eve,*nextve; | 
					
						
							|  |  |  | 	EditEdge *eed,*nexted; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	addlisttolist(tempve,&fillvertbase); | 
					
						
							|  |  |  | 	addlisttolist(temped,&filledgebase); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	eve= tempve->first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		nextve= eve->next; | 
					
						
							|  |  |  | 		if(eve->xs==nr) { | 
					
						
							|  |  |  | 			BLI_remlink(tempve,eve); | 
					
						
							|  |  |  | 			BLI_addtail(&fillvertbase,eve); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= nextve; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	eed= temped->first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		nexted= eed->next; | 
					
						
							|  |  |  | 		if(eed->f1==nr) { | 
					
						
							|  |  |  | 			BLI_remlink(temped,eed); | 
					
						
							|  |  |  | 			BLI_addtail(&filledgebase,eed); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= nexted; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | static void scanfill(PolyFill *pf, int mat_nr) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	ScFillVert *sc = NULL, *sc1; | 
					
						
							|  |  |  | 	EditVert *eve,*v1,*v2,*v3; | 
					
						
							|  |  |  | 	EditEdge *eed,*nexted,*ed1,*ed2,*ed3; | 
					
						
							|  |  |  | 	float miny = 0.0; | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 	int a,b,verts, maxface, totface;	 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	short nr, test, twoconnected=0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nr= pf->nr; | 
					
						
							|  |  |  | 	verts= pf->verts; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* PRINTS
 | 
					
						
							|  |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		printf("vert: %x co: %f %f\n",eve,eve->co[cox],eve->co[coy]); | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	}	 | 
					
						
							|  |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		printf("edge: %x  verts: %x %x\n",eed,eed->v1,eed->v2); | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 0: remove zero sized edges */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		if(eed->v1->co[cox]==eed->v2->co[cox]) { | 
					
						
							|  |  |  | 			if(eed->v1->co[coy]==eed->v2->co[coy]) { | 
					
						
							|  |  |  | 				if(eed->v1->f==255 && eed->v2->f!=255) { | 
					
						
							|  |  |  | 					eed->v2->f= 255; | 
					
						
							| 
									
										
										
										
											2006-01-03 17:22:42 +00:00
										 |  |  | 					eed->v2->tmp.v= eed->v1->tmp.v; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else if(eed->v2->f==255 && eed->v1->f!=255) { | 
					
						
							|  |  |  | 					eed->v1->f= 255; | 
					
						
							| 
									
										
										
										
											2006-01-03 17:22:42 +00:00
										 |  |  | 					eed->v1->tmp.v= eed->v2->tmp.v; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else if(eed->v2->f==255 && eed->v1->f==255) { | 
					
						
							| 
									
										
										
										
											2006-01-03 17:22:42 +00:00
										 |  |  | 					eed->v1->tmp.v= eed->v2->tmp.v; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					eed->v2->f= 255; | 
					
						
							| 
									
										
										
										
											2006-06-30 16:07:50 +00:00
										 |  |  | 					eed->v2->tmp.v = eed->v1->tmp.v; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 1: make using FillVert and FillEdge lists a sorted
 | 
					
						
							|  |  |  | 		ScFillVert list | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 	sc= scdata= (ScFillVert *)MEM_callocN(pf->verts*sizeof(ScFillVert),"Scanfill1"); | 
					
						
							|  |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	verts= 0; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->xs==nr) { | 
					
						
							|  |  |  | 			if(eve->f!= 255) { | 
					
						
							|  |  |  | 				verts++; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				eve->f= 0;	/* flag for connectedges later on */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				sc->v1= eve; | 
					
						
							|  |  |  | 				sc++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qsort(scdata, verts, sizeof(ScFillVert), vergscdata); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sc= scdata; | 
					
						
							|  |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		nexted= eed->next; | 
					
						
							|  |  |  | 		eed->f= 0; | 
					
						
							|  |  |  | 		BLI_remlink(&filledgebase,eed); | 
					
						
							| 
									
										
										
										
											2006-06-30 16:07:50 +00:00
										 |  |  | /* commented all of this out, this I have no idea for what it is for, probably from ancient past */ | 
					
						
							|  |  |  | /* it does crash blender, since it uses mixed original and new vertices (ton) */ | 
					
						
							|  |  |  | //		if(eed->v1->f==255) {
 | 
					
						
							|  |  |  | //			v1= eed->v1;
 | 
					
						
							|  |  |  | //			while((eed->v1->f == 255) && (eed->v1->tmp.v != v1)) 
 | 
					
						
							|  |  |  | //				eed->v1 = eed->v1->tmp.v;
 | 
					
						
							|  |  |  | //		}
 | 
					
						
							|  |  |  | //		if(eed->v2->f==255) {
 | 
					
						
							|  |  |  | //			v2= eed->v2;
 | 
					
						
							|  |  |  | //			while((eed->v2->f == 255) && (eed->v2->tmp.v != v2))
 | 
					
						
							|  |  |  | //				eed->v2 = eed->v2->tmp.v;
 | 
					
						
							|  |  |  | //		}
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		if(eed->v1!=eed->v2) addedgetoscanlist(eed,verts); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		eed= nexted; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	sc= scdata; | 
					
						
							|  |  |  | 	for(a=0;a<verts;a++) { | 
					
						
							|  |  |  | 		printf("\nscvert: %x\n",sc->v1); | 
					
						
							|  |  |  | 		eed= sc->first; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			printf(" ed %x %x %x\n",eed,eed->v1,eed->v2); | 
					
						
							|  |  |  | 			eed= eed->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		sc++; | 
					
						
							|  |  |  | 	}*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 2: FILL LOOP */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(pf->f==0) twoconnected= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* (temporal) security: never much more faces than vertices */ | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 	totface= 0; | 
					
						
							|  |  |  | 	maxface= 2*verts;		/* 2*verts: based at a filled circle within a triangle */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	sc= scdata; | 
					
						
							|  |  |  | 	for(a=0;a<verts;a++) { | 
					
						
							|  |  |  | 		/* printf("VERTEX %d %x\n",a,sc->v1); */ | 
					
						
							|  |  |  | 		ed1= sc->first; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 		while(ed1) {	/* set connectflags  */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			nexted= ed1->next; | 
					
						
							|  |  |  | 			if(ed1->v1->h==1 || ed1->v2->h==1) { | 
					
						
							|  |  |  | 				BLI_remlink((ListBase *)&(sc->first),ed1); | 
					
						
							|  |  |  | 				BLI_addtail(&filledgebase,ed1); | 
					
						
							|  |  |  | 				if(ed1->v1->h>1) ed1->v1->h--; | 
					
						
							|  |  |  | 				if(ed1->v2->h>1) ed1->v2->h--; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else ed1->v2->f= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ed1= nexted; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 		while(sc->first) {	/* for as long there are edges */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			ed1= sc->first; | 
					
						
							|  |  |  | 			ed2= ed1->next; | 
					
						
							| 
									
										
										
										
											2004-10-30 22:53:27 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			/* commented out... the ESC here delivers corrupted memory (and doesnt work during grab) */ | 
					
						
							|  |  |  | 			/* if(callLocalInterruptCallBack()) break; */ | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 			if(totface>maxface) { | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				/* printf("Fill error: endless loop. Escaped at vert %d,  tot: %d.\n", a, verts); */ | 
					
						
							|  |  |  | 				a= verts; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(ed2==0) { | 
					
						
							|  |  |  | 				sc->first=sc->last= 0; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				/* printf("just 1 edge to vert\n"); */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				BLI_addtail(&filledgebase,ed1); | 
					
						
							|  |  |  | 				ed1->v2->f= 0; | 
					
						
							|  |  |  | 				ed1->v1->h--;  | 
					
						
							|  |  |  | 				ed1->v2->h--; | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				/* test rest of vertices */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				v1= ed1->v2; | 
					
						
							|  |  |  | 				v2= ed1->v1; | 
					
						
							|  |  |  | 				v3= ed2->v2; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				/* this happens with a serial of overlapping edges */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				if(v1==v2 || v2==v3) break; | 
					
						
							|  |  |  | 				/* printf("test verts %x %x %x\n",v1,v2,v3); */ | 
					
						
							|  |  |  | 				miny = ( (v1->co[coy])<(v3->co[coy]) ? (v1->co[coy]) : (v3->co[coy]) ); | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				/*  miny= MIN2(v1->co[coy],v3->co[coy]); */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				sc1= sc+1; | 
					
						
							|  |  |  | 				test= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				for(b=a+1;b<verts;b++) { | 
					
						
							|  |  |  | 					if(sc1->v1->f==0) { | 
					
						
							|  |  |  | 						if(sc1->v1->co[coy] <= miny) break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if(testedgeside(v1->co,v2->co,sc1->v1->co)) | 
					
						
							|  |  |  | 							if(testedgeside(v2->co,v3->co,sc1->v1->co)) | 
					
						
							|  |  |  | 								if(testedgeside(v3->co,v1->co,sc1->v1->co)) { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 									/* point in triangle */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 								 | 
					
						
							|  |  |  | 									test= 1; | 
					
						
							|  |  |  | 									break; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					sc1++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if(test) { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 					/* make new edge, and start over */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 					/* printf("add new edge %x %x and start again\n",v2,sc1->v1); */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					ed3= BLI_addfilledge(v2, sc1->v1); | 
					
						
							|  |  |  | 					BLI_remlink(&filledgebase, ed3); | 
					
						
							|  |  |  | 					BLI_insertlinkbefore((ListBase *)&(sc->first), ed2, ed3); | 
					
						
							|  |  |  | 					ed3->v2->f= 1; | 
					
						
							|  |  |  | 					ed3->f= 2; | 
					
						
							|  |  |  | 					ed3->v1->h++;  | 
					
						
							|  |  |  | 					ed3->v2->h++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 					/* new triangle */ | 
					
						
							|  |  |  | 					/* printf("add face %x %x %x\n",v1,v2,v3); */ | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | 					addfillface(v1, v2, v3, mat_nr); | 
					
						
							| 
									
										
										
										
											2004-09-18 20:15:37 +00:00
										 |  |  | 					totface++; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 					BLI_remlink((ListBase *)&(sc->first),ed1); | 
					
						
							|  |  |  | 					BLI_addtail(&filledgebase,ed1); | 
					
						
							|  |  |  | 					ed1->v2->f= 0; | 
					
						
							|  |  |  | 					ed1->v1->h--;  | 
					
						
							|  |  |  | 					ed1->v2->h--; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 					/* ed2 can be removed when it's an old one */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 					if(ed2->f==0 && twoconnected) { | 
					
						
							|  |  |  | 						BLI_remlink((ListBase *)&(sc->first),ed2); | 
					
						
							|  |  |  | 						BLI_addtail(&filledgebase,ed2); | 
					
						
							|  |  |  | 						ed2->v2->f= 0; | 
					
						
							|  |  |  | 						ed2->v1->h--;  | 
					
						
							|  |  |  | 						ed2->v2->h--; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 					/* new edge */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 					ed3= BLI_addfilledge(v1, v3); | 
					
						
							|  |  |  | 					BLI_remlink(&filledgebase, ed3); | 
					
						
							|  |  |  | 					ed3->f= 2; | 
					
						
							|  |  |  | 					ed3->v1->h++;  | 
					
						
							|  |  |  | 					ed3->v2->h++; | 
					
						
							|  |  |  | 					 | 
					
						
							|  |  |  | 					/* printf("add new edge %x %x\n",v1,v3); */ | 
					
						
							|  |  |  | 					sc1= addedgetoscanlist(ed3, verts); | 
					
						
							|  |  |  | 					 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 					if(sc1) {	/* ed3 already exists: remove */ | 
					
						
							|  |  |  | 						/* printf("Edge exists\n"); */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 						ed3->v1->h--;  | 
					
						
							|  |  |  | 						ed3->v2->h--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if(twoconnected) ed3= sc1->first; | 
					
						
							|  |  |  | 						else ed3= 0; | 
					
						
							|  |  |  | 						while(ed3) { | 
					
						
							|  |  |  | 							if( (ed3->v1==v1 && ed3->v2==v3) || (ed3->v1==v3 && ed3->v2==v1) ) { | 
					
						
							|  |  |  | 								BLI_remlink((ListBase *)&(sc1->first),ed3); | 
					
						
							|  |  |  | 								BLI_addtail(&filledgebase,ed3); | 
					
						
							|  |  |  | 								ed3->v1->h--;  | 
					
						
							|  |  |  | 								ed3->v2->h--; | 
					
						
							|  |  |  | 								break; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							ed3= ed3->next; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 			/* test for loose edges */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			ed1= sc->first; | 
					
						
							|  |  |  | 			while(ed1) { | 
					
						
							|  |  |  | 				nexted= ed1->next; | 
					
						
							|  |  |  | 				if(ed1->v1->h<2 || ed1->v2->h<2) { | 
					
						
							|  |  |  | 					BLI_remlink((ListBase *)&(sc->first),ed1); | 
					
						
							|  |  |  | 					BLI_addtail(&filledgebase,ed1); | 
					
						
							|  |  |  | 					if(ed1->v1->h>1) ed1->v1->h--; | 
					
						
							|  |  |  | 					if(ed1->v2->h>1) ed1->v2->h--; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				ed1= nexted; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		sc++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(scdata); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-15 05:18:45 +00:00
										 |  |  | int BLI_edgefill(int mat_nr) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/*
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	  - fill works with its own lists, so create that first (no faces!) | 
					
						
							| 
									
										
										
										
											2006-01-03 17:22:42 +00:00
										 |  |  | 	  - for vertices, put in ->tmp.v the old pointer | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	  - struct elements xs en ys are not used here: don't hide stuff in it | 
					
						
							|  |  |  | 	  - edge flag ->f becomes 2 when it's a new edge | 
					
						
							|  |  |  | 	  - mode: & 1 is check for crossings, then create edges (TO DO ) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 	ListBase tempve, temped; | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	EditEdge *eed,*nexted; | 
					
						
							|  |  |  | 	PolyFill *pflist,*pf; | 
					
						
							|  |  |  | 	float *minp, *maxp, *v1, *v2, norm[3], len; | 
					
						
							|  |  |  | 	short a,c,poly=0,ok=0,toggle=0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* reset variables */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		eve->f= 0; | 
					
						
							|  |  |  | 		eve->xs= 0; | 
					
						
							|  |  |  | 		eve->h= 0; | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* first test vertices if they are in edges */ | 
					
						
							|  |  |  | 	/* including resetting of flags */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		eed->f= eed->f1= eed->h= 0; | 
					
						
							|  |  |  | 		eed->v1->f= 1; | 
					
						
							|  |  |  | 		eed->v2->f= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->f & 1) { | 
					
						
							|  |  |  | 			ok=1;  | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ok==0) return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* NEW NEW! define projection: with 'best' normal */ | 
					
						
							|  |  |  | 	/* just use the first three different vertices */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* THIS PART STILL IS PRETTY WEAK! (ton) */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	eve= fillvertbase.last; | 
					
						
							|  |  |  | 	len= 0.0; | 
					
						
							|  |  |  | 	v1= eve->co; | 
					
						
							|  |  |  | 	v2= 0; | 
					
						
							|  |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(v2) { | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 			if( compare_v3v3(v2, eve->co, COMPLIMIT)==0) { | 
					
						
							|  |  |  | 				len= normal_tri_v3( norm,v1, v2, eve->co); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				if(len != 0.0) break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | 		else if(compare_v3v3(v1, eve->co, COMPLIMIT)==0) { | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			v2= eve->co; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	if(len==0.0) return 0;	/* no fill possible */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	norm[0]= fabs(norm[0]); | 
					
						
							|  |  |  | 	norm[1]= fabs(norm[1]); | 
					
						
							|  |  |  | 	norm[2]= fabs(norm[2]); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(norm[2]>=norm[0] && norm[2]>=norm[1]) { | 
					
						
							|  |  |  | 		cox= 0; coy= 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(norm[1]>=norm[0] && norm[1]>=norm[2]) { | 
					
						
							|  |  |  | 		cox= 0; coy= 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		cox= 1; coy= 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 1: COUNT POLYS */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 		/* get first vertex with no poly number */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		if(eve->xs==0) { | 
					
						
							|  |  |  | 			poly++; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 			/* now a sortof select connected */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			ok= 1; | 
					
						
							|  |  |  | 			eve->xs= poly; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			while(ok) { | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				ok= 0; | 
					
						
							|  |  |  | 				toggle++; | 
					
						
							|  |  |  | 				if(toggle & 1) eed= filledgebase.first; | 
					
						
							|  |  |  | 				else eed= filledgebase.last; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				while(eed) { | 
					
						
							|  |  |  | 					if(eed->v1->xs==0 && eed->v2->xs==poly) { | 
					
						
							|  |  |  | 						eed->v1->xs= poly; | 
					
						
							|  |  |  | 						eed->f1= poly; | 
					
						
							|  |  |  | 						ok= 1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if(eed->v2->xs==0 && eed->v1->xs==poly) { | 
					
						
							|  |  |  | 						eed->v2->xs= poly; | 
					
						
							|  |  |  | 						eed->f1= poly; | 
					
						
							|  |  |  | 						ok= 1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if(eed->f1==0) { | 
					
						
							|  |  |  | 						if(eed->v1->xs==poly && eed->v2->xs==poly) { | 
					
						
							|  |  |  | 							eed->f1= poly; | 
					
						
							|  |  |  | 							ok= 1; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if(toggle & 1) eed= eed->next; | 
					
						
							|  |  |  | 					else eed= eed->prev; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* printf("amount of poly's: %d\n",poly); */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 2: remove loose edges and strings of edges */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		if(eed->v1->h++ >250) break; | 
					
						
							|  |  |  | 		if(eed->v2->h++ >250) break; | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(eed) { | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 		/* otherwise it's impossible to be sure you can clear vertices */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		callLocalErrorCallBack("No vertices with 250 edges allowed!"); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* does it only for vertices with ->h==1 */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	testvertexnearedge(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ok= 1; | 
					
						
							|  |  |  | 	while(ok) { | 
					
						
							|  |  |  | 		ok= 0; | 
					
						
							|  |  |  | 		toggle++; | 
					
						
							|  |  |  | 		if(toggle & 1) eed= filledgebase.first; | 
					
						
							|  |  |  | 		else eed= filledgebase.last; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			if(toggle & 1) nexted= eed->next; | 
					
						
							|  |  |  | 			else nexted= eed->prev; | 
					
						
							|  |  |  | 			if(eed->v1->h==1) { | 
					
						
							|  |  |  | 				eed->v2->h--; | 
					
						
							|  |  |  | 				BLI_remlink(&fillvertbase,eed->v1);  | 
					
						
							|  |  |  | 				BLI_remlink(&filledgebase,eed);  | 
					
						
							|  |  |  | 				ok= 1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if(eed->v2->h==1) { | 
					
						
							|  |  |  | 				eed->v1->h--; | 
					
						
							|  |  |  | 				BLI_remlink(&fillvertbase,eed->v2);  | 
					
						
							|  |  |  | 				BLI_remlink(&filledgebase,eed);  | 
					
						
							|  |  |  | 				ok= 1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eed= nexted; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(filledgebase.first==0) { | 
					
						
							|  |  |  | 		/* printf("All edges removed\n"); */ | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* CURRENT STATUS:
 | 
					
						
							| 
									
										
										
										
											2006-01-03 17:22:42 +00:00
										 |  |  | 	- eve->f      :1= availalble in edges | 
					
						
							|  |  |  | 	- eve->xs     :polynumber | 
					
						
							|  |  |  | 	- eve->h      :amount of edges connected to vertex | 
					
						
							|  |  |  | 	- eve->tmp.v  :store! original vertex number | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	- eed->f  : | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	- eed->f1 :poly number | 
					
						
							|  |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 3: MAKE POLYFILL STRUCT */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	pflist= (PolyFill *)MEM_callocN(poly*sizeof(PolyFill),"edgefill"); | 
					
						
							|  |  |  | 	pf= pflist; | 
					
						
							|  |  |  | 	for(a=1;a<=poly;a++) { | 
					
						
							|  |  |  | 		pf->nr= a; | 
					
						
							|  |  |  | 		pf->min[0]=pf->min[1]=pf->min[2]= 1.0e20; | 
					
						
							|  |  |  | 		pf->max[0]=pf->max[1]=pf->max[2]= -1.0e20; | 
					
						
							|  |  |  | 		pf++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	eed= filledgebase.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		pflist[eed->f1-1].edges++; | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	eve= fillvertbase.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		pflist[eve->xs-1].verts++; | 
					
						
							|  |  |  | 		minp= pflist[eve->xs-1].min; | 
					
						
							|  |  |  | 		maxp= pflist[eve->xs-1].max; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		minp[cox]= (minp[cox])<(eve->co[cox]) ? (minp[cox]) : (eve->co[cox]); | 
					
						
							|  |  |  | 		minp[coy]= (minp[coy])<(eve->co[coy]) ? (minp[coy]) : (eve->co[coy]); | 
					
						
							|  |  |  | 		maxp[cox]= (maxp[cox])>(eve->co[cox]) ? (maxp[cox]) : (eve->co[cox]); | 
					
						
							|  |  |  | 		maxp[coy]= (maxp[coy])>(eve->co[coy]) ? (maxp[coy]) : (eve->co[coy]); | 
					
						
							|  |  |  | 		if(eve->h>2) pflist[eve->xs-1].f= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 4: FIND HOLES OR BOUNDS, JOIN THEM
 | 
					
						
							|  |  |  | 	 *  ( bounds just to divide it in pieces for optimization,  | 
					
						
							|  |  |  | 	 *    the edgefill itself has good auto-hole detection) | 
					
						
							|  |  |  | 	 * WATCH IT: ONLY WORKS WITH SORTED POLYS!!! */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if(poly>1) { | 
					
						
							|  |  |  | 		short *polycache, *pc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 		/* so, sort first */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		qsort(pflist, poly, sizeof(PolyFill), vergpoly); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/*pf= pflist;
 | 
					
						
							|  |  |  | 		for(a=1;a<=poly;a++) { | 
					
						
							|  |  |  | 			printf("poly:%d edges:%d verts:%d flag: %d\n",a,pf->edges,pf->verts,pf->f); | 
					
						
							|  |  |  | 			PRINT2(f, f, pf->min[0], pf->min[1]); | 
					
						
							|  |  |  | 			pf++; | 
					
						
							|  |  |  | 		}*/ | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 		polycache= pc= MEM_callocN(sizeof(short)*poly, "polycache"); | 
					
						
							|  |  |  | 		pf= pflist; | 
					
						
							|  |  |  | 		for(a=0; a<poly; a++, pf++) { | 
					
						
							|  |  |  | 			for(c=a+1;c<poly;c++) { | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				/* if 'a' inside 'c': join (bbox too)
 | 
					
						
							|  |  |  | 				 * Careful: 'a' can also be inside another poly. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				 */ | 
					
						
							|  |  |  | 				if(boundisect(pf, pflist+c)) { | 
					
						
							|  |  |  | 					*pc= c; | 
					
						
							|  |  |  | 					pc++; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 				/* only for optimize! */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				/* else if(pf->max[cox] < (pflist+c)->min[cox]) break; */ | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			while(pc!=polycache) { | 
					
						
							|  |  |  | 				pc--; | 
					
						
							|  |  |  | 				mergepolysSimp(pf, pflist+ *pc); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		MEM_freeN(polycache); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	pf= pflist; | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* printf("after merge\n");
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	for(a=1;a<=poly;a++) { | 
					
						
							|  |  |  | 		printf("poly:%d edges:%d verts:%d flag: %d\n",a,pf->edges,pf->verts,pf->f); | 
					
						
							|  |  |  | 		pf++; | 
					
						
							|  |  |  | 	} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* STEP 5: MAKE TRIANGLES */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	tempve.first= fillvertbase.first; | 
					
						
							|  |  |  | 	tempve.last= fillvertbase.last; | 
					
						
							|  |  |  | 	temped.first= filledgebase.first; | 
					
						
							|  |  |  | 	temped.last= filledgebase.last; | 
					
						
							|  |  |  | 	fillvertbase.first=fillvertbase.last= 0; | 
					
						
							|  |  |  | 	filledgebase.first=filledgebase.last= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pf= pflist; | 
					
						
							|  |  |  | 	for(a=0;a<poly;a++) { | 
					
						
							|  |  |  | 		if(pf->edges>1) { | 
					
						
							|  |  |  | 			splitlist(&tempve,&temped,pf->nr); | 
					
						
							| 
									
										
										
										
											2005-08-14 06:08:41 +00:00
										 |  |  | 			scanfill(pf, mat_nr); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		pf++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	addlisttolist(&fillvertbase,&tempve); | 
					
						
							|  |  |  | 	addlisttolist(&filledgebase,&temped); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 16:02:26 +00:00
										 |  |  | 	/* FREE */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(pflist); | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |