| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  |  /* $Id:
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2004 by Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Johnny Matthews, Geoffrey Bantle. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | editmesh_tool.c: UI called tools for editmesh, geometry changes here, otherwise in mods.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <math.h>
 | 
					
						
							|  |  |  | #include <float.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | #include "PIL_time.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLO_sys_types.h" // for intptr_t support
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							|  |  |  | #include "DNA_material_types.h"
 | 
					
						
							|  |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							|  |  |  | #include "DNA_modifier_types.h"
 | 
					
						
							|  |  |  | #include "DNA_object_types.h"
 | 
					
						
							|  |  |  | #include "DNA_scene_types.h"
 | 
					
						
							|  |  |  | #include "DNA_screen_types.h"
 | 
					
						
							|  |  |  | #include "DNA_view3d_types.h"
 | 
					
						
							|  |  |  | #include "DNA_key_types.h"
 | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | #include "DNA_windowmanager_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "RNA_types.h"
 | 
					
						
							|  |  |  | #include "RNA_define.h"
 | 
					
						
							|  |  |  | #include "RNA_access.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_blenlib.h"
 | 
					
						
							|  |  |  | #include "BLI_arithb.h"
 | 
					
						
							|  |  |  | #include "BLI_editVert.h"
 | 
					
						
							|  |  |  | #include "BLI_rand.h"
 | 
					
						
							|  |  |  | #include "BLI_ghash.h"
 | 
					
						
							|  |  |  | #include "BLI_linklist.h"
 | 
					
						
							|  |  |  | #include "BLI_heap.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | #include "BKE_context.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "BKE_customdata.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | #include "BKE_depsgraph.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "BKE_global.h"
 | 
					
						
							|  |  |  | #include "BKE_library.h"
 | 
					
						
							|  |  |  | #include "BKE_mesh.h"
 | 
					
						
							|  |  |  | #include "BKE_object.h"
 | 
					
						
							|  |  |  | #include "BKE_utildefines.h"
 | 
					
						
							|  |  |  | #include "BKE_bmesh.h"
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | #include "BKE_report.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BIF_gl.h"
 | 
					
						
							|  |  |  | #include "BIF_glutil.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | #include "WM_api.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | #include "WM_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "ED_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | #include "ED_screen.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-08 16:17:47 +00:00
										 |  |  | #include "ED_transform.h"
 | 
					
						
							|  |  |  | #include "ED_util.h"
 | 
					
						
							|  |  |  | #include "ED_view3d.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 15:46:35 +00:00
										 |  |  | #include "UI_interface.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-01 13:15:35 +00:00
										 |  |  | #include "mesh_intern.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* XXX */ | 
					
						
							|  |  |  | static int extern_qread() {return 0;} | 
					
						
							| 
									
										
										
										
											2009-07-31 01:40:15 +00:00
										 |  |  | static void waitcursor(int val) {} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | static int pupmenu() {return 0;} | 
					
						
							|  |  |  | static int qtest() {return 0;} | 
					
						
							|  |  |  | #define add_numbut(a, b, c, d, e, f, g) {}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* XXX */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* local prototypes ---------------*/ | 
					
						
							|  |  |  | static void free_tagged_edges_faces(EditMesh *em, EditEdge *eed, EditFace *efa); | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | int EdgeLoopDelete(EditMesh *em, wmOperator *op); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /********* qsort routines *********/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct xvertsort { | 
					
						
							|  |  |  | 	float x; | 
					
						
							|  |  |  | 	EditVert *v1; | 
					
						
							|  |  |  | } xvertsort; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int vergxco(const void *v1, const void *v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const xvertsort *x1=v1, *x2=v2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if( x1->x > x2->x ) return 1; | 
					
						
							|  |  |  | 	else if( x1->x < x2->x) return -1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct facesort { | 
					
						
							|  |  |  | 	uintptr_t x; | 
					
						
							|  |  |  | 	struct EditFace *efa; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int vergface(const void *v1, const void *v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const struct facesort *x1=v1, *x2=v2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if( x1->x > x2->x ) return 1; | 
					
						
							|  |  |  | 	else if( x1->x < x2->x) return -1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* *********************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void convert_to_triface(EditMesh *em, int direction) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *efa, *efan, *next; | 
					
						
							|  |  |  | 	float fac; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	efa= em->faces.last; | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		next= efa->prev; | 
					
						
							|  |  |  | 		if(efa->v4) { | 
					
						
							|  |  |  | 			if(efa->f & SELECT) { | 
					
						
							|  |  |  | 				/* choose shortest diagonal for split */ | 
					
						
							|  |  |  | 				fac= VecLenf(efa->v1->co, efa->v3->co) - VecLenf(efa->v2->co, efa->v4->co); | 
					
						
							|  |  |  | 				/* this makes sure exact squares get split different in both cases */ | 
					
						
							|  |  |  | 				if( (direction==0 && fac<FLT_EPSILON) || (direction && fac>0.0f) ) { | 
					
						
							|  |  |  | 					efan= EM_face_from_faces(em, efa, NULL, 0, 1, 2, -1); | 
					
						
							|  |  |  | 					if(efa->f & SELECT) EM_select_face(efan, 1); | 
					
						
							|  |  |  | 					efan= EM_face_from_faces(em, efa, NULL, 0, 2, 3, -1); | 
					
						
							|  |  |  | 					if(efa->f & SELECT) EM_select_face(efan, 1); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					efan= EM_face_from_faces(em, efa, NULL, 0, 1, 3, -1); | 
					
						
							|  |  |  | 					if(efa->f & SELECT) EM_select_face(efan, 1); | 
					
						
							|  |  |  | 					efan= EM_face_from_faces(em, efa, NULL, 1, 2, 3, -1); | 
					
						
							|  |  |  | 					if(efa->f & SELECT) EM_select_face(efan, 1); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 				free_editface(em, efa); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		efa= next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EM_fgon_flags(em);	// redo flags and indices for fgons
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int removedoublesflag(EditMesh *em, short flag, short automerge, float limit)		/* return amount */ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 		flag -		Test with vert->flags | 
					
						
							|  |  |  | 		automerge -	Alternative operation, merge unselected into selected. | 
					
						
							|  |  |  | 					Used for "Auto Weld" mode. warning. | 
					
						
							|  |  |  | 		limit -		Quick manhattan distance between verts. | 
					
						
							|  |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* all verts with (flag & 'flag') are being evaluated */ | 
					
						
							|  |  |  | 	EditVert *eve, *v1, *nextve; | 
					
						
							|  |  |  | 	EditEdge *eed, *e1, *nexted; | 
					
						
							|  |  |  | 	EditFace *efa, *nextvl; | 
					
						
							|  |  |  | 	xvertsort *sortblock, *sb, *sb1; | 
					
						
							|  |  |  | 	struct facesort *vlsortblock, *vsb, *vsb1; | 
					
						
							|  |  |  | 	int a, b, test, amount; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* flag 128 is cleared, count */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Normal non weld operation */ | 
					
						
							|  |  |  | 	eve= em->verts.first; | 
					
						
							|  |  |  | 	amount= 0; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		eve->f &= ~128; | 
					
						
							|  |  |  | 		if(eve->h==0 && (automerge || (eve->f & flag))) amount++; | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(amount==0) return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* allocate memory and qsort */ | 
					
						
							|  |  |  | 	sb= sortblock= MEM_mallocN(sizeof(xvertsort)*amount,"sortremovedoub"); | 
					
						
							|  |  |  | 	eve= em->verts.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->h==0 && (automerge || (eve->f & flag))) { | 
					
						
							|  |  |  | 			sb->x= eve->co[0]+eve->co[1]+eve->co[2]; | 
					
						
							|  |  |  | 			sb->v1= eve; | 
					
						
							|  |  |  | 			sb++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	qsort(sortblock, amount, sizeof(xvertsort), vergxco); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* test for doubles */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	sb= sortblock; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (automerge) { | 
					
						
							|  |  |  | 		for(a=0; a<amount; a++, sb++) { | 
					
						
							|  |  |  | 			eve= sb->v1; | 
					
						
							|  |  |  | 			if( (eve->f & 128)==0 ) { | 
					
						
							|  |  |  | 				sb1= sb+1; | 
					
						
							|  |  |  | 				for(b=a+1; b<amount && (eve->f & 128)==0; b++, sb1++) { | 
					
						
							|  |  |  | 					if(sb1->x - sb->x > limit) break; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					/* when automarge, only allow unselected->selected */ | 
					
						
							|  |  |  | 					v1= sb1->v1; | 
					
						
							|  |  |  | 					if( (v1->f & 128)==0 ) { | 
					
						
							|  |  |  | 						if ((eve->f & flag)==0 && (v1->f & flag)==1) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							if(	(float)fabs(v1->co[0]-eve->co[0])<=limit && | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								(float)fabs(v1->co[1]-eve->co[1])<=limit && | 
					
						
							|  |  |  | 								(float)fabs(v1->co[2]-eve->co[2])<=limit) | 
					
						
							|  |  |  | 							{	/* unique bit */ | 
					
						
							|  |  |  | 								eve->f|= 128; | 
					
						
							|  |  |  | 								eve->tmp.v = v1; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} else if(	(eve->f & flag)==1 && (v1->f & flag)==0 ) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							if(	(float)fabs(v1->co[0]-eve->co[0])<=limit && | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								(float)fabs(v1->co[1]-eve->co[1])<=limit && | 
					
						
							|  |  |  | 								(float)fabs(v1->co[2]-eve->co[2])<=limit) | 
					
						
							|  |  |  | 							{	/* unique bit */ | 
					
						
							|  |  |  | 								v1->f|= 128; | 
					
						
							|  |  |  | 								v1->tmp.v = eve; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		for(a=0; a<amount; a++, sb++) { | 
					
						
							|  |  |  | 			eve= sb->v1; | 
					
						
							|  |  |  | 			if( (eve->f & 128)==0 ) { | 
					
						
							|  |  |  | 				sb1= sb+1; | 
					
						
							|  |  |  | 				for(b=a+1; b<amount; b++, sb1++) { | 
					
						
							|  |  |  | 					/* first test: simpel dist */ | 
					
						
							|  |  |  | 					if(sb1->x - sb->x > limit) break; | 
					
						
							|  |  |  | 					v1= sb1->v1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					/* second test: is vertex allowed */ | 
					
						
							|  |  |  | 					if( (v1->f & 128)==0 ) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						if(	(float)fabs(v1->co[0]-eve->co[0])<=limit && | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							(float)fabs(v1->co[1]-eve->co[1])<=limit && | 
					
						
							|  |  |  | 							(float)fabs(v1->co[2]-eve->co[2])<=limit) | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							v1->f|= 128; | 
					
						
							|  |  |  | 							v1->tmp.v = eve; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	MEM_freeN(sortblock); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (!automerge) | 
					
						
							|  |  |  | 		for(eve = em->verts.first; eve; eve=eve->next) | 
					
						
							|  |  |  | 			if((eve->f & flag) && (eve->f & 128)) | 
					
						
							|  |  |  | 				EM_data_interp_from_verts(em, eve, eve->tmp.v, eve->tmp.v, 0.5f); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* test edges and insert again */ | 
					
						
							|  |  |  | 	eed= em->edges.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		eed->f2= 0; | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	eed= em->edges.last; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		nexted= eed->prev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(eed->f2==0) { | 
					
						
							|  |  |  | 			if( (eed->v1->f & 128) || (eed->v2->f & 128) ) { | 
					
						
							|  |  |  | 				remedge(em, eed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(eed->v1->f & 128) eed->v1 = eed->v1->tmp.v; | 
					
						
							|  |  |  | 				if(eed->v2->f & 128) eed->v2 = eed->v2->tmp.v; | 
					
						
							|  |  |  | 				e1= addedgelist(em, eed->v1, eed->v2, eed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(e1) { | 
					
						
							|  |  |  | 					e1->f2= 1; | 
					
						
							|  |  |  | 					if(eed->f & SELECT) | 
					
						
							|  |  |  | 						e1->f |= SELECT; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if(e1!=eed) free_editedge(em, eed); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= nexted; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* first count amount of test faces */ | 
					
						
							|  |  |  | 	efa= (struct EditFace *)em->faces.first; | 
					
						
							|  |  |  | 	amount= 0; | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		efa->f1= 0; | 
					
						
							|  |  |  | 		if(efa->v1->f & 128) efa->f1= 1; | 
					
						
							|  |  |  | 		else if(efa->v2->f & 128) efa->f1= 1; | 
					
						
							|  |  |  | 		else if(efa->v3->f & 128) efa->f1= 1; | 
					
						
							|  |  |  | 		else if(efa->v4 && (efa->v4->f & 128)) efa->f1= 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(efa->f1==1) amount++; | 
					
						
							|  |  |  | 		efa= efa->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* test faces for double vertices, and if needed remove them */ | 
					
						
							|  |  |  | 	efa= (struct EditFace *)em->faces.first; | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		nextvl= efa->next; | 
					
						
							|  |  |  | 		if(efa->f1==1) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(efa->v1->f & 128) efa->v1= efa->v1->tmp.v; | 
					
						
							|  |  |  | 			if(efa->v2->f & 128) efa->v2= efa->v2->tmp.v; | 
					
						
							|  |  |  | 			if(efa->v3->f & 128) efa->v3= efa->v3->tmp.v; | 
					
						
							|  |  |  | 			if(efa->v4 && (efa->v4->f & 128)) efa->v4= efa->v4->tmp.v; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			test= 0; | 
					
						
							|  |  |  | 			if(efa->v1==efa->v2) test+=1; | 
					
						
							|  |  |  | 			if(efa->v2==efa->v3) test+=2; | 
					
						
							|  |  |  | 			if(efa->v3==efa->v1) test+=4; | 
					
						
							|  |  |  | 			if(efa->v4==efa->v1) test+=8; | 
					
						
							|  |  |  | 			if(efa->v3==efa->v4) test+=16; | 
					
						
							|  |  |  | 			if(efa->v2==efa->v4) test+=32; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(test) { | 
					
						
							|  |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					if(test==1 || test==2) { | 
					
						
							|  |  |  | 						efa->v2= efa->v3; | 
					
						
							|  |  |  | 						efa->v3= efa->v4; | 
					
						
							|  |  |  | 						efa->v4= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						EM_data_interp_from_faces(em, efa, NULL, efa, 0, 2, 3, 3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						test= 0; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if(test==8 || test==16) { | 
					
						
							|  |  |  | 						efa->v4= 0; | 
					
						
							|  |  |  | 						test= 0; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 						free_editface(em, efa); | 
					
						
							|  |  |  | 						amount--; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 					free_editface(em, efa); | 
					
						
							|  |  |  | 					amount--; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(test==0) { | 
					
						
							|  |  |  | 				/* set edge pointers */ | 
					
						
							|  |  |  | 				efa->e1= findedgelist(em, efa->v1, efa->v2); | 
					
						
							|  |  |  | 				efa->e2= findedgelist(em, efa->v2, efa->v3); | 
					
						
							|  |  |  | 				if(efa->v4==0) { | 
					
						
							|  |  |  | 					efa->e3= findedgelist(em, efa->v3, efa->v1); | 
					
						
							|  |  |  | 					efa->e4= 0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					efa->e3= findedgelist(em, efa->v3, efa->v4); | 
					
						
							|  |  |  | 					efa->e4= findedgelist(em, efa->v4, efa->v1); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		efa= nextvl; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* double faces: sort block */ | 
					
						
							|  |  |  | 	/* count again, now all selected faces */ | 
					
						
							|  |  |  | 	amount= 0; | 
					
						
							|  |  |  | 	efa= em->faces.first; | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		efa->f1= 0; | 
					
						
							|  |  |  | 		if(faceselectedOR(efa, 1)) { | 
					
						
							|  |  |  | 			efa->f1= 1; | 
					
						
							|  |  |  | 			amount++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		efa= efa->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(amount) { | 
					
						
							|  |  |  | 		/* double faces: sort block */ | 
					
						
							|  |  |  | 		vsb= vlsortblock= MEM_mallocN(sizeof(struct facesort)*amount, "sortremovedoub"); | 
					
						
							|  |  |  | 		efa= em->faces.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			if(efa->f1 & 1) { | 
					
						
							|  |  |  | 				if(efa->v4) vsb->x= (uintptr_t) MIN4( (uintptr_t)efa->v1, (uintptr_t)efa->v2, (uintptr_t)efa->v3, (uintptr_t)efa->v4); | 
					
						
							|  |  |  | 				else vsb->x= (uintptr_t) MIN3( (uintptr_t)efa->v1, (uintptr_t)efa->v2, (uintptr_t)efa->v3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				vsb->efa= efa; | 
					
						
							|  |  |  | 				vsb++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			efa= efa->next; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		qsort(vlsortblock, amount, sizeof(struct facesort), vergface); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		vsb= vlsortblock; | 
					
						
							|  |  |  | 		for(a=0; a<amount; a++) { | 
					
						
							|  |  |  | 			efa= vsb->efa; | 
					
						
							|  |  |  | 			if( (efa->f1 & 128)==0 ) { | 
					
						
							|  |  |  | 				vsb1= vsb+1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				for(b=a+1; b<amount; b++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					/* first test: same pointer? */ | 
					
						
							|  |  |  | 					if(vsb->x != vsb1->x) break; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					/* second test: is test permitted? */ | 
					
						
							|  |  |  | 					efa= vsb1->efa; | 
					
						
							|  |  |  | 					if( (efa->f1 & 128)==0 ) { | 
					
						
							|  |  |  | 						if( compareface(efa, vsb->efa)) efa->f1 |= 128; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					vsb1++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			vsb++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		MEM_freeN(vlsortblock); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/* remove double faces */ | 
					
						
							|  |  |  | 		efa= (struct EditFace *)em->faces.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			nextvl= efa->next; | 
					
						
							|  |  |  | 			if(efa->f1 & 128) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 				free_editface(em, efa); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			efa= nextvl; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* remove double vertices */ | 
					
						
							|  |  |  | 	a= 0; | 
					
						
							|  |  |  | 	eve= (struct EditVert *)em->verts.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		nextve= eve->next; | 
					
						
							|  |  |  | 		if(automerge || eve->f & flag) { | 
					
						
							|  |  |  | 			if(eve->f & 128) { | 
					
						
							|  |  |  | 				a++; | 
					
						
							|  |  |  | 				BLI_remlink(&em->verts, eve); | 
					
						
							|  |  |  | 				free_editvert(em, eve); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= nextve; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a;	/* amount */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-23 03:07:07 +00:00
										 |  |  | static int removedoublesflag_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh(((Mesh *)obedit->data)); | 
					
						
							| 
									
										
										
										
											2009-09-17 14:46:22 +00:00
										 |  |  | 	/*char msg[100];
 | 
					
						
							| 
									
										
										
										
											2009-01-23 03:07:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-17 14:46:22 +00:00
										 |  |  | 	int cnt = removedoublesflag(em,1,0,RNA_float_get(op->ptr, "limit"));*/ | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 06:06:01 +00:00
										 |  |  | 	/*XXX this messes up last operator panel
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 	if(cnt) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		sprintf(msg, "Removed %d vertices", cnt); | 
					
						
							|  |  |  | 		BKE_report(op->reports, RPT_INFO, msg); | 
					
						
							| 
									
										
										
										
											2009-09-14 06:06:01 +00:00
										 |  |  | 	}*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2009-01-23 03:07:07 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | void MESH_OT_remove_doubles(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-23 03:07:07 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Remove Doubles"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Remove duplicate vertices."; | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | 	ot->idname= "MESH_OT_remove_doubles"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-23 03:07:07 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= removedoublesflag_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-09-14 06:06:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	RNA_def_float(ot->srna, "limit", 0.00001f, 0.000001f, 50.0f, "Merge Threshold", "Minimum distance between merged verts", 0.00001f, 10.0f); | 
					
						
							| 
									
										
										
										
											2009-01-23 03:07:07 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // XXX is this needed?
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* called from buttons */ | 
					
						
							|  |  |  | static void xsortvert_flag__doSetX(void *userData, EditVert *eve, int x, int y, int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	xvertsort *sortblock = userData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sortblock[index].x = x; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | /* all verts with (flag & 'flag') are sorted */ | 
					
						
							|  |  |  | void xsortvert_flag(bContext *C, int flag) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	ViewContext vc; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	xvertsort *sortblock; | 
					
						
							|  |  |  | 	ListBase tbase; | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	int i, amount; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	em_setup_viewcontext(C, &vc); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	amount = BLI_countlist(&vc.em->verts); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	sortblock = MEM_callocN(sizeof(xvertsort)*amount,"xsort"); | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	for (i=0,eve= vc.em->verts.first; eve; i++,eve=eve->next) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eve->f & flag) | 
					
						
							|  |  |  | 			sortblock[i].v1 = eve; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	mesh_foreachScreenVert(&vc, xsortvert_flag__doSetX, sortblock, 0); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	qsort(sortblock, amount, sizeof(xvertsort), vergxco); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/* make temporal listbase */ | 
					
						
							|  |  |  | 	tbase.first= tbase.last= 0; | 
					
						
							|  |  |  | 	for (i=0; i<amount; i++) { | 
					
						
							|  |  |  | 		eve = sortblock[i].v1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (eve) { | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 			BLI_remlink(&vc.em->verts, eve); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			BLI_addtail(&tbase, eve); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 16:03:29 +00:00
										 |  |  | 	addlisttolist(&vc.em->verts, &tbase); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MEM_freeN(sortblock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* called from buttons */ | 
					
						
							|  |  |  | void hashvert_flag(EditMesh *em, int flag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* switch vertex order using hash table */ | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	struct xvertsort *sortblock, *sb, onth, *newsort; | 
					
						
							|  |  |  | 	ListBase tbase; | 
					
						
							|  |  |  | 	int amount, a, b; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* count */ | 
					
						
							|  |  |  | 	eve= em->verts.first; | 
					
						
							|  |  |  | 	amount= 0; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->f & flag) amount++; | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(amount==0) return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* allocate memory */ | 
					
						
							|  |  |  | 	sb= sortblock= (struct xvertsort *)MEM_mallocN(sizeof(struct xvertsort)*amount,"sortremovedoub"); | 
					
						
							|  |  |  | 	eve= em->verts.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->f & flag) { | 
					
						
							|  |  |  | 			sb->v1= eve; | 
					
						
							|  |  |  | 			sb++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_srand(1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	sb= sortblock; | 
					
						
							|  |  |  | 	for(a=0; a<amount; a++, sb++) { | 
					
						
							|  |  |  | 		b= (int)(amount*BLI_drand()); | 
					
						
							|  |  |  | 		if(b>=0 && b<amount) { | 
					
						
							|  |  |  | 			newsort= sortblock+b; | 
					
						
							|  |  |  | 			onth= *sb; | 
					
						
							|  |  |  | 			*sb= *newsort; | 
					
						
							|  |  |  | 			*newsort= onth; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* make temporal listbase */ | 
					
						
							|  |  |  | 	tbase.first= tbase.last= 0; | 
					
						
							|  |  |  | 	sb= sortblock; | 
					
						
							|  |  |  | 	while(amount--) { | 
					
						
							|  |  |  | 		eve= sb->v1; | 
					
						
							|  |  |  | 		BLI_remlink(&em->verts, eve); | 
					
						
							|  |  |  | 		BLI_addtail(&tbase, eve); | 
					
						
							|  |  |  | 		sb++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	addlisttolist(&em->verts, &tbase); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MEM_freeN(sortblock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* generic extern called extruder */ | 
					
						
							| 
									
										
										
										
											2009-07-19 04:50:10 +00:00
										 |  |  | void extrude_mesh(Scene *scene, Object *obedit, EditMesh *em, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float nor[3]= {0.0, 0.0, 0.0}; | 
					
						
							|  |  |  | 	short nr, transmode= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-19 17:14:26 +00:00
										 |  |  | 	/* extrude depends on totvertsel etc */ | 
					
						
							|  |  |  | 	EM_stats_update(em); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(em->selectmode & SCE_SELECT_VERTEX) { | 
					
						
							| 
									
										
										
										
											2009-01-15 15:01:39 +00:00
										 |  |  | 		if(em->totvertsel==0) nr= 0; | 
					
						
							|  |  |  | 		else if(em->totvertsel==1) nr= 4; | 
					
						
							|  |  |  | 		else if(em->totedgesel==0) nr= 4; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		else if(em->totfacesel==0) | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			nr= 3; // pupmenu("Extrude %t|Only Edges%x3|Only Vertices%x4");
 | 
					
						
							| 
									
										
										
										
											2009-01-15 15:01:39 +00:00
										 |  |  | 		else if(em->totfacesel==1) | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			nr= 1; // pupmenu("Extrude %t|Region %x1|Only Edges%x3|Only Vertices%x4");
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			nr= 1; // pupmenu("Extrude %t|Region %x1||Individual Faces %x2|Only Edges%x3|Only Vertices%x4");
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else if(em->selectmode & SCE_SELECT_EDGE) { | 
					
						
							| 
									
										
										
										
											2009-01-15 15:01:39 +00:00
										 |  |  | 		if (em->totedgesel==0) nr = 0; | 
					
						
							|  |  |  | 		else if (em->totedgesel==1) nr = 3; | 
					
						
							|  |  |  | 		else if(em->totfacesel==0) nr = 3; | 
					
						
							|  |  |  | 		else if(em->totfacesel==1) | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			nr= 1; // pupmenu("Extrude %t|Region %x1|Only Edges%x3");
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			nr= 1; // pupmenu("Extrude %t|Region %x1||Individual Faces %x2|Only Edges%x3");
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2009-01-15 15:01:39 +00:00
										 |  |  | 		if (em->totfacesel == 0) nr = 0; | 
					
						
							|  |  |  | 		else if (em->totfacesel == 1) nr = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			nr= 1; // pupmenu("Extrude %t|Region %x1||Individual Faces %x2");
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(nr<1) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	if(nr==1)  transmode= extrudeflag(obedit, em, SELECT, nor); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	else if(nr==4) transmode= extrudeflag_verts_indiv(em, SELECT, nor); | 
					
						
							|  |  |  | 	else if(nr==3) transmode= extrudeflag_edges_indiv(em, SELECT, nor); | 
					
						
							|  |  |  | 	else transmode= extrudeflag_face_indiv(em, SELECT, nor); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(transmode==0) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Not a valid selection for extrude"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		EM_fgon_flags(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/* We need to force immediate calculation here because
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			* transform may use derived objects (which are now stale). | 
					
						
							|  |  |  | 			* | 
					
						
							|  |  |  | 			* This shouldn't be necessary, derived queries should be | 
					
						
							|  |  |  | 			* automatically building this data if invalid. Or something. | 
					
						
							|  |  |  | 			*/ | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | //		DAG_id_flush_update(obedit->data, OB_RECALC_DATA);
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 		object_handle_update(scene, obedit); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* individual faces? */ | 
					
						
							|  |  |  | //		BIF_TransformSetUndo("Extrude");
 | 
					
						
							|  |  |  | 		if(nr==2) { | 
					
						
							|  |  |  | //			initTransform(TFM_SHRINKFATTEN, CTX_NO_PET|CTX_NO_MIRROR);
 | 
					
						
							|  |  |  | //			Transform();
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | //			initTransform(TFM_TRANSLATION, CTX_NO_PET|CTX_NO_MIRROR);
 | 
					
						
							|  |  |  | 			if(transmode=='n') { | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 				Mat4MulVecfl(obedit->obmat, nor); | 
					
						
							|  |  |  | 				VecSubf(nor, nor, obedit->obmat[3]); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | //				BIF_setSingleAxisConstraint(nor, "along normal");
 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | //			Transform();
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | // XXX should be a menu item
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | static int mesh_extrude_invoke(bContext *C, wmOperator *op, wmEvent *event) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	Scene *scene= CTX_data_scene(C); | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
											  
											
												2.5
Operator goodies!
--- Macro operators
Operators now can consist of multiple operators. Such a macro operator
is identical and behaves identical to other opererators. Macros can
also be constructed of macros even! Currently only hardcoded macros are
implemented, this to solve combined operators such as 'add duplicate' or
'extrude' (both want a transform appended).
Usage is simple:
- WM_operatortype_append_macro() : add new operatortype, name, flags
- WM_operatortype_macro_define() : add existing operator to macro
(Note: macro_define will also allow properties to be set, doesnt work
 right now)
On converting the macro wmOperatorType to a real operator, it makes a
list of all operators, and the standard macro callbacks (exec, invoke,
modal, poll) just will use all.
Important note; switching to a modal operator only works as last in the
chain now!
Macros implemented for duplicate, extrude and rip. Tool menu works fine
for it, also the redo hotkey F4 works properly.
--- Operator redo fix
The operators use the undo system to switch back, but this could give
errors if other actions added undo pushes (buttons, outliner). Now the
redo for operator searches back for the correct undo level.
This fixes issues with many redos.
Note for brecht: removed the ED_undo_push for buttons... it was called
on *every* button now, which is probably too much? For example, using
the 'toolbar' redo also caused this...
											
										 
											2009-07-29 17:56:38 +00:00
										 |  |  | //	int constraint_axis[3] = {0, 0, 1};
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-19 04:50:10 +00:00
										 |  |  | 	extrude_mesh(scene, obedit, em, op); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	RNA_enum_set(op->ptr, "proportional", 0); | 
					
						
							|  |  |  | 	RNA_boolean_set(op->ptr, "mirror", 0); | 
					
						
							| 
									
										
										
										
											2009-07-16 03:16:03 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* the following two should only be set when extruding faces */ | 
					
						
							| 
									
										
											  
											
												2.5
Operator goodies!
--- Macro operators
Operators now can consist of multiple operators. Such a macro operator
is identical and behaves identical to other opererators. Macros can
also be constructed of macros even! Currently only hardcoded macros are
implemented, this to solve combined operators such as 'add duplicate' or
'extrude' (both want a transform appended).
Usage is simple:
- WM_operatortype_append_macro() : add new operatortype, name, flags
- WM_operatortype_macro_define() : add existing operator to macro
(Note: macro_define will also allow properties to be set, doesnt work
 right now)
On converting the macro wmOperatorType to a real operator, it makes a
list of all operators, and the standard macro callbacks (exec, invoke,
modal, poll) just will use all.
Important note; switching to a modal operator only works as last in the
chain now!
Macros implemented for duplicate, extrude and rip. Tool menu works fine
for it, also the redo hotkey F4 works properly.
--- Operator redo fix
The operators use the undo system to switch back, but this could give
errors if other actions added undo pushes (buttons, outliner). Now the
redo for operator searches back for the correct undo level.
This fixes issues with many redos.
Note for brecht: removed the ED_undo_push for buttons... it was called
on *every* button now, which is probably too much? For example, using
the 'toolbar' redo also caused this...
											
										 
											2009-07-29 17:56:38 +00:00
										 |  |  | //	RNA_enum_set(op->ptr, "constraint_orientation", V3D_MANIP_NORMAL);
 | 
					
						
							|  |  |  | //	RNA_boolean_set_array(op->ptr, "constraint_axis", constraint_axis);
 | 
					
						
							| 
									
										
										
										
											2009-07-16 03:16:03 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
											  
											
												2.5
Operator goodies!
--- Macro operators
Operators now can consist of multiple operators. Such a macro operator
is identical and behaves identical to other opererators. Macros can
also be constructed of macros even! Currently only hardcoded macros are
implemented, this to solve combined operators such as 'add duplicate' or
'extrude' (both want a transform appended).
Usage is simple:
- WM_operatortype_append_macro() : add new operatortype, name, flags
- WM_operatortype_macro_define() : add existing operator to macro
(Note: macro_define will also allow properties to be set, doesnt work
 right now)
On converting the macro wmOperatorType to a real operator, it makes a
list of all operators, and the standard macro callbacks (exec, invoke,
modal, poll) just will use all.
Important note; switching to a modal operator only works as last in the
chain now!
Macros implemented for duplicate, extrude and rip. Tool menu works fine
for it, also the redo hotkey F4 works properly.
--- Operator redo fix
The operators use the undo system to switch back, but this could give
errors if other actions added undo pushes (buttons, outliner). Now the
redo for operator searches back for the correct undo level.
This fixes issues with many redos.
Note for brecht: removed the ED_undo_push for buttons... it was called
on *every* button now, which is probably too much? For example, using
the 'toolbar' redo also caused this...
											
										 
											2009-07-29 17:56:38 +00:00
										 |  |  | //	WM_operator_name_call(C, "TFM_OT_translate", WM_OP_INVOKE_REGION_WIN, op->ptr);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* extrude without transform */ | 
					
						
							|  |  |  | static int mesh_extrude_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	Scene *scene= CTX_data_scene(C); | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh(obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-19 04:50:10 +00:00
										 |  |  | 	extrude_mesh(scene, obedit, em, op); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | void MESH_OT_extrude(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ot->name= "Extrude"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Extrude selected vertices, edges or faces."; | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | 	ot->idname= "MESH_OT_extrude"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	ot->invoke= mesh_extrude_invoke; | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | 	ot->exec= mesh_extrude_exec; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int split_mesh(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	WM_cursor_wait(1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* make duplicate first */ | 
					
						
							|  |  |  | 	adduplicateflag(em, SELECT); | 
					
						
							|  |  |  | 	/* old faces have flag 128 set, delete them */ | 
					
						
							|  |  |  | 	delfaceflag(em, 128); | 
					
						
							|  |  |  | 	recalc_editnormals(em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	WM_cursor_wait(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-29 02:15:13 +00:00
										 |  |  | void MESH_OT_split(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ot->name= "Split"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Split selected geometry into separate disconnected mesh."; | 
					
						
							| 
									
										
										
										
											2009-03-29 02:15:13 +00:00
										 |  |  | 	ot->idname= "MESH_OT_split"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= split_mesh; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int extrude_repeat_mesh(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-03-30 07:28:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	RegionView3D *rv3d = ED_view3d_context_rv3d(C); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	int steps = RNA_int_get(op->ptr,"steps"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	float offs = RNA_float_get(op->ptr,"offset"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	float dvec[3], tmat[3][3], bmat[3][3], nor[3]= {0.0, 0.0, 0.0}; | 
					
						
							|  |  |  | 	short a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* dvec */ | 
					
						
							| 
									
										
											  
											
												2.5
View3D has been split now in a local part (RegionView3D) and a
per-area part (old View3D). Currently local is:
- view transform
- camera zoom/offset
- gpencil (todo)
- custom clipping planes
Rest is in Area still, like active camera, draw type, layers,
localview, custom centers, around-settings, transform widget,
gridlines, and so on (mostly stuff as available in header).
To see it work; also added new feature for region split, 
press SHIFT+ALT+CTRL+S for four-split. 
The idea is to make a preset 4-split, configured to stick
to top/right/front views for three views.
Another cool idea to explore is to then box-clip all drawing
based on these 3 views.
Note about the code:
- currently view3d still stores some depricated settings, to
  convert from older files. Not all settings are copied over
  though, like custom clip planes or the 'lock view to object'.
- since some view3d ops are now on area level, the operators
  for it should keep track of that.
Bugfix in transform: quat initialize in operator-invoke missed
one zero.
Als brought back GE to compile for missing Ipos and channels.
											
										 
											2009-01-19 16:54:41 +00:00
										 |  |  | 	dvec[0]= rv3d->persinv[2][0]; | 
					
						
							|  |  |  | 	dvec[1]= rv3d->persinv[2][1]; | 
					
						
							|  |  |  | 	dvec[2]= rv3d->persinv[2][2]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Normalize(dvec); | 
					
						
							|  |  |  | 	dvec[0]*= offs; | 
					
						
							|  |  |  | 	dvec[1]*= offs; | 
					
						
							|  |  |  | 	dvec[2]*= offs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* base correction */ | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	Mat3CpyMat4(bmat, obedit->obmat); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Mat3Inv(tmat, bmat); | 
					
						
							|  |  |  | 	Mat3MulVecfl(tmat, dvec); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(a=0; a<steps; a++) { | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 		extrudeflag(obedit, em, SELECT, nor); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		translateflag(em, SELECT, dvec); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	recalc_editnormals(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EM_fgon_flags(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | void MESH_OT_extrude_repeat(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Extrude Repeat Mesh"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Extrude selected vertices, edges or faces repeatedly."; | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	ot->idname= "MESH_OT_extrude_repeat"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= extrude_repeat_mesh; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* props */ | 
					
						
							|  |  |  | 	RNA_def_float(ot->srna, "offset", 2.0f, 0.0f, 100.0f, "Offset", "", 0.0f, FLT_MAX); | 
					
						
							|  |  |  | 	RNA_def_int(ot->srna, "steps", 10, 0, 180, "Steps", "", 0, INT_MAX); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ************************** spin operator ******************** */ | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int spin_mesh(bContext *C, wmOperator *op, float *dvec, int steps, float degr, int dupli ) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	ToolSettings *ts= CTX_data_tool_settings(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditVert *eve,*nextve; | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	float nor[3]= {0.0f, 0.0f, 0.0f}; | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	float si, n[3], q[4], cmat[3][3], imat[3][3], tmat[3][3]; | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	float cent[3], bmat[3][3]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	float phi; | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	short a, ok= 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	RNA_float_get_array(op->ptr, "center", cent); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* imat and center and size */ | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	Mat3CpyMat4(bmat, obedit->obmat); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Mat3Inv(imat,bmat); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	cent[0]-= obedit->obmat[3][0]; | 
					
						
							|  |  |  | 	cent[1]-= obedit->obmat[3][1]; | 
					
						
							|  |  |  | 	cent[2]-= obedit->obmat[3][2]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Mat3MulVecfl(imat, cent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	phi= degr*M_PI/360.0; | 
					
						
							|  |  |  | 	phi/= steps; | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	if(ts->editbutflag & B_CLOCKWISE) phi= -phi; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	RNA_float_get_array(op->ptr, "axis", n); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Normalize(n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	q[0]= (float)cos(phi); | 
					
						
							|  |  |  | 	si= (float)sin(phi); | 
					
						
							|  |  |  | 	q[1]= n[0]*si; | 
					
						
							|  |  |  | 	q[2]= n[1]*si; | 
					
						
							|  |  |  | 	q[3]= n[2]*si; | 
					
						
							|  |  |  | 	QuatToMat3(q, cmat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Mat3MulMat3(tmat,cmat,bmat); | 
					
						
							|  |  |  | 	Mat3MulMat3(bmat,imat,tmat); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(dupli==0) | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 		if(ts->editbutflag & B_KEEPORIG) | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 			adduplicateflag(em, 1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	for(a=0; a<steps; a++) { | 
					
						
							|  |  |  | 		if(dupli==0) ok= extrudeflag(obedit, em, SELECT, nor); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		else adduplicateflag(em, SELECT); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 		if(ok==0) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		rotateflag(em, SELECT, cent, bmat); | 
					
						
							|  |  |  | 		if(dvec) { | 
					
						
							|  |  |  | 			Mat3MulVecfl(bmat,dvec); | 
					
						
							|  |  |  | 			translateflag(em, SELECT, dvec); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ok==0) { | 
					
						
							|  |  |  | 		/* no vertices or only loose ones selected, remove duplicates */ | 
					
						
							|  |  |  | 		eve= em->verts.first; | 
					
						
							|  |  |  | 		while(eve) { | 
					
						
							|  |  |  | 			nextve= eve->next; | 
					
						
							|  |  |  | 			if(eve->f & SELECT) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->verts,eve); | 
					
						
							|  |  |  | 				free_editvert(em, eve); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eve= nextve; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		recalc_editnormals(em); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 		EM_fgon_flags(em); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 		DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	return ok; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | static int spin_mesh_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	int ok; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	ok= spin_mesh(C, op, NULL, RNA_int_get(op->ptr,"steps"), RNA_float_get(op->ptr,"degrees"), RNA_boolean_get(op->ptr,"dupli")); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	if(ok==0) { | 
					
						
							|  |  |  | 		BKE_report(op->reports, RPT_ERROR, "No valid vertices are selected"); | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | /* get center and axis, in global coords */ | 
					
						
							|  |  |  | static int spin_mesh_invoke(bContext *C, wmOperator *op, wmEvent *event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Scene *scene = CTX_data_scene(C); | 
					
						
							|  |  |  | 	View3D *v3d = CTX_wm_view3d(C); | 
					
						
							| 
									
										
										
										
											2009-06-26 15:48:09 +00:00
										 |  |  | 	RegionView3D *rv3d= ED_view3d_context_rv3d(C); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	RNA_float_set_array(op->ptr, "center", give_cursor(scene, v3d)); | 
					
						
							|  |  |  | 	RNA_float_set_array(op->ptr, "axis", rv3d->viewinv[2]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	return spin_mesh_exec(C, op); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | void MESH_OT_spin(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Spin"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Extrude selected vertices in a circle around the cursor in indicated viewport."; | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	ot->idname= "MESH_OT_spin"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	ot->invoke= spin_mesh_invoke; | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	ot->exec= spin_mesh_exec; | 
					
						
							| 
									
										
										
										
											2009-06-16 13:09:36 +00:00
										 |  |  | 	ot->poll= EM_view3d_poll; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	/* props */ | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	RNA_def_int(ot->srna, "steps", 9, 0, INT_MAX, "Steps", "Steps", 0, INT_MAX); | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "dupli", 0, "Dupli", "Make Duplicates"); | 
					
						
							|  |  |  | 	RNA_def_float(ot->srna, "degrees", 90.0f, -FLT_MAX, FLT_MAX, "Degrees", "Degrees", -360.0f, 360.0f); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	RNA_def_float_vector(ot->srna, "center", 3, NULL, -FLT_MAX, FLT_MAX, "Center", "Center in global view space", -FLT_MAX, FLT_MAX); | 
					
						
							|  |  |  | 	RNA_def_float_vector(ot->srna, "axis", 3, NULL, -1.0f, 1.0f, "Axis", "Axis in global view space", -FLT_MAX, FLT_MAX); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | static int screw_mesh_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-16 20:04:01 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditVert *eve,*v1=0,*v2=0; | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	float dvec[3], nor[3]; | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	int steps, turns; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	turns= RNA_int_get(op->ptr, "turns"); | 
					
						
							|  |  |  | 	steps= RNA_int_get(op->ptr, "steps"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* clear flags */ | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	for(eve= em->verts.first; eve; eve= eve->next) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		eve->f1= 0; | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* edges set flags in verts */ | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	for(eed= em->edges.first; eed; eed= eed->next) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eed->v1->f & SELECT) { | 
					
						
							|  |  |  | 			if(eed->v2->f & SELECT) { | 
					
						
							|  |  |  | 				/* watch: f1 is a byte */ | 
					
						
							|  |  |  | 				if(eed->v1->f1<2) eed->v1->f1++; | 
					
						
							|  |  |  | 				if(eed->v2->f1<2) eed->v2->f1++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* find two vertices with eve->f1==1, more or less is wrong */ | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	for(eve= em->verts.first; eve; eve= eve->next) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eve->f1==1) { | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 			if(v1==NULL) v1= eve; | 
					
						
							|  |  |  | 			else if(v2==NULL) v2= eve; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 				v1= NULL; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	if(v1==NULL || v2==NULL) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "You have to select a string of connected vertices too"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* calculate dvec */ | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	dvec[0]= ( v1->co[0]- v2->co[0] )/steps; | 
					
						
							|  |  |  | 	dvec[1]= ( v1->co[1]- v2->co[1] )/steps; | 
					
						
							|  |  |  | 	dvec[2]= ( v1->co[2]- v2->co[2] )/steps; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	VECCOPY(nor, obedit->obmat[2]); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(nor[0]*dvec[0]+nor[1]*dvec[1]+nor[2]*dvec[2]>0.000) { | 
					
						
							|  |  |  | 		dvec[0]= -dvec[0]; | 
					
						
							|  |  |  | 		dvec[1]= -dvec[1]; | 
					
						
							|  |  |  | 		dvec[2]= -dvec[2]; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	if(spin_mesh(C, op, dvec, turns*steps, 360.0f*turns, 0)) { | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 		DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 		WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 		return OPERATOR_FINISHED; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		BKE_report(op->reports, RPT_ERROR, "No valid vertices are selected"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | /* get center and axis, in global coords */ | 
					
						
							|  |  |  | static int screw_mesh_invoke(bContext *C, wmOperator *op, wmEvent *event) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Scene *scene = CTX_data_scene(C); | 
					
						
							|  |  |  | 	View3D *v3d = CTX_wm_view3d(C); | 
					
						
							| 
									
										
										
										
											2009-06-26 15:48:09 +00:00
										 |  |  | 	RegionView3D *rv3d= ED_view3d_context_rv3d(C); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	RNA_float_set_array(op->ptr, "center", give_cursor(scene, v3d)); | 
					
						
							|  |  |  | 	RNA_float_set_array(op->ptr, "axis", rv3d->viewinv[1]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	return screw_mesh_exec(C, op); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | void MESH_OT_screw(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Screw"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Extrude selected vertices in screw-shaped rotation around the cursor in indicated viewport."; | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | 	ot->idname= "MESH_OT_screw"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 	ot->invoke= screw_mesh_invoke; | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | 	ot->exec= screw_mesh_exec; | 
					
						
							| 
									
										
										
										
											2009-06-16 13:09:36 +00:00
										 |  |  | 	ot->poll= EM_view3d_poll; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | 	/*props */ | 
					
						
							| 
									
										
										
										
											2009-02-19 16:22:07 +00:00
										 |  |  | 	RNA_def_int(ot->srna, "steps", 9, 0, INT_MAX, "Steps", "Steps", 0, 256); | 
					
						
							|  |  |  | 	RNA_def_int(ot->srna, "turns", 1, 0, INT_MAX, "Turns", "Turns", 0, 256); | 
					
						
							| 
									
										
										
										
											2009-06-01 14:08:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	RNA_def_float_vector(ot->srna, "center", 3, NULL, -FLT_MAX, FLT_MAX, "Center", "Center in global view space", -FLT_MAX, FLT_MAX); | 
					
						
							|  |  |  | 	RNA_def_float_vector(ot->srna, "axis", 3, NULL, -1.0f, 1.0f, "Axis", "Axis in global view space", -FLT_MAX, FLT_MAX); | 
					
						
							| 
									
										
										
										
											2009-02-18 03:01:45 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void erase_edges(EditMesh *em, ListBase *l) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *ed, *nexted; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	ed = (EditEdge *) l->first; | 
					
						
							|  |  |  | 	while(ed) { | 
					
						
							|  |  |  | 		nexted= ed->next; | 
					
						
							|  |  |  | 		if( (ed->v1->f & SELECT) || (ed->v2->f & SELECT) ) { | 
					
						
							|  |  |  | 			remedge(em, ed); | 
					
						
							|  |  |  | 			free_editedge(em, ed); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ed= nexted; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void erase_faces(EditMesh *em, ListBase *l) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *f, *nextf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	f = (EditFace *) l->first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(f) { | 
					
						
							|  |  |  | 		nextf= f->next; | 
					
						
							|  |  |  | 		if( faceselectedOR(f, SELECT) ) { | 
					
						
							|  |  |  | 			BLI_remlink(l, f); | 
					
						
							|  |  |  | 			free_editface(em, f); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		f = nextf; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void erase_vertices(EditMesh *em, ListBase *l) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *v, *nextv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	v = (EditVert *) l->first; | 
					
						
							|  |  |  | 	while(v) { | 
					
						
							|  |  |  | 		nextv= v->next; | 
					
						
							|  |  |  | 		if(v->f & 1) { | 
					
						
							|  |  |  | 			BLI_remlink(l, v); | 
					
						
							|  |  |  | 			free_editvert(em, v); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		v = nextv; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | void delete_mesh(Object *obedit, EditMesh *em, wmOperator *op, int event) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditFace *efa, *nextvl; | 
					
						
							|  |  |  | 	EditVert *eve,*nextve; | 
					
						
							|  |  |  | 	EditEdge *eed,*nexted; | 
					
						
							|  |  |  | 	int count; | 
					
						
							|  |  |  | 	char *str="Erase"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(event<1) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(event==10 ) { | 
					
						
							|  |  |  | 		str= "Erase Vertices"; | 
					
						
							|  |  |  | 		erase_edges(em, &em->edges); | 
					
						
							|  |  |  | 		erase_faces(em, &em->faces); | 
					
						
							|  |  |  | 		erase_vertices(em, &em->verts); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	else if(event==6) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		if(!EdgeLoopDelete(em, op)) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		str= "Erase Edge Loop"; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(event==4) { | 
					
						
							|  |  |  | 		str= "Erase Edges & Faces"; | 
					
						
							|  |  |  | 		efa= em->faces.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			nextvl= efa->next; | 
					
						
							|  |  |  | 			/* delete only faces with 1 or more edges selected */ | 
					
						
							|  |  |  | 			count= 0; | 
					
						
							|  |  |  | 			if(efa->e1->f & SELECT) count++; | 
					
						
							|  |  |  | 			if(efa->e2->f & SELECT) count++; | 
					
						
							|  |  |  | 			if(efa->e3->f & SELECT) count++; | 
					
						
							|  |  |  | 			if(efa->e4 && (efa->e4->f & SELECT)) count++; | 
					
						
							|  |  |  | 			if(count) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 				free_editface(em, efa); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			efa= nextvl; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= em->edges.first; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			nexted= eed->next; | 
					
						
							|  |  |  | 			if(eed->f & SELECT) { | 
					
						
							|  |  |  | 				remedge(em, eed); | 
					
						
							|  |  |  | 				free_editedge(em, eed); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eed= nexted; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		efa= em->faces.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			nextvl= efa->next; | 
					
						
							|  |  |  | 			event=0; | 
					
						
							|  |  |  | 			if( efa->v1->f & SELECT) event++; | 
					
						
							|  |  |  | 			if( efa->v2->f & SELECT) event++; | 
					
						
							|  |  |  | 			if( efa->v3->f & SELECT) event++; | 
					
						
							|  |  |  | 			if(efa->v4 && (efa->v4->f & SELECT)) event++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(event>1) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 				free_editface(em, efa); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			efa= nextvl; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	else if(event==1) { | 
					
						
							|  |  |  | 		str= "Erase Edges"; | 
					
						
							|  |  |  | 		// faces first
 | 
					
						
							|  |  |  | 		efa= em->faces.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			nextvl= efa->next; | 
					
						
							|  |  |  | 			event=0; | 
					
						
							|  |  |  | 			if( efa->e1->f & SELECT) event++; | 
					
						
							|  |  |  | 			if( efa->e2->f & SELECT) event++; | 
					
						
							|  |  |  | 			if( efa->e3->f & SELECT) event++; | 
					
						
							|  |  |  | 			if(efa->e4 && (efa->e4->f & SELECT)) event++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(event) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 				free_editface(em, efa); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			efa= nextvl; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= em->edges.first; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			nexted= eed->next; | 
					
						
							|  |  |  | 			if(eed->f & SELECT) { | 
					
						
							|  |  |  | 				remedge(em, eed); | 
					
						
							|  |  |  | 				free_editedge(em, eed); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eed= nexted; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		/* to remove loose vertices: */ | 
					
						
							|  |  |  | 		eed= em->edges.first; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			if( eed->v1->f & SELECT) eed->v1->f-=SELECT; | 
					
						
							|  |  |  | 			if( eed->v2->f & SELECT) eed->v2->f-=SELECT; | 
					
						
							|  |  |  | 			eed= eed->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= em->verts.first; | 
					
						
							|  |  |  | 		while(eve) { | 
					
						
							|  |  |  | 			nextve= eve->next; | 
					
						
							|  |  |  | 			if(eve->f & SELECT) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->verts,eve); | 
					
						
							|  |  |  | 				free_editvert(em, eve); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eve= nextve; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(event==2) { | 
					
						
							|  |  |  | 		str="Erase Faces"; | 
					
						
							|  |  |  | 		delfaceflag(em, SELECT); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(event==3) { | 
					
						
							|  |  |  | 		str= "Erase All"; | 
					
						
							|  |  |  | 		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)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(event==5) { | 
					
						
							|  |  |  | 		str= "Erase Only Faces"; | 
					
						
							|  |  |  | 		efa= em->faces.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			nextvl= efa->next; | 
					
						
							|  |  |  | 			if(efa->f & SELECT) { | 
					
						
							|  |  |  | 				BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 				free_editface(em, efa); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			efa= nextvl; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	EM_fgon_flags(em);	// redo flags and indices for fgons
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 02:37:12 +00:00
										 |  |  | /* Note, these values must match delete_mesh() event values */ | 
					
						
							|  |  |  | static EnumPropertyItem prop_mesh_delete_types[] = { | 
					
						
							| 
									
										
										
										
											2009-06-16 00:52:21 +00:00
										 |  |  | 	{10,"VERT",		0, "Vertices", ""}, | 
					
						
							|  |  |  | 	{1, "EDGE",		0, "Edges", ""}, | 
					
						
							|  |  |  | 	{2, "FACE",		0, "Faces", ""}, | 
					
						
							|  |  |  | 	{3, "ALL",		0, "All", ""}, | 
					
						
							|  |  |  | 	{4, "EDGE_FACE",0, "Edges & Faces", ""}, | 
					
						
							|  |  |  | 	{5, "ONLY_FACE",0, "Only Faces", ""}, | 
					
						
							|  |  |  | 	{6, "EDGE_LOOP",0, "Edge Loop", ""}, | 
					
						
							|  |  |  | 	{0, NULL, 0, NULL, NULL} | 
					
						
							| 
									
										
										
										
											2009-02-01 02:37:12 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | static int delete_mesh_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	delete_mesh(obedit, em, op, RNA_enum_get(op->ptr, "type")); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | void MESH_OT_delete(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2009-02-11 23:02:21 +00:00
										 |  |  | 	ot->name= "Delete"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Delete selected vertices, edges or faces."; | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | 	ot->idname= "MESH_OT_delete"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-02-01 02:37:12 +00:00
										 |  |  | 	ot->invoke= WM_menu_invoke; | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | 	ot->exec= delete_mesh_exec; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | 	/*props */ | 
					
						
							| 
									
										
										
										
											2009-02-01 02:37:12 +00:00
										 |  |  | 	RNA_def_enum(ot->srna, "type", prop_mesh_delete_types, 10, "Type", "Method used for deleting mesh data"); | 
					
						
							| 
									
										
										
										
											2009-01-31 02:31:58 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /*GB*/ | 
					
						
							|  |  |  | /*-------------------------------------------------------------------------------*/ | 
					
						
							|  |  |  | /*--------------------------- Edge Based Subdivide ------------------------------*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define EDGENEW	2
 | 
					
						
							|  |  |  | #define FACENEW	2
 | 
					
						
							|  |  |  | #define EDGEINNER  4
 | 
					
						
							|  |  |  | #define EDGEOLD  8
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*used by faceloop cut to select only edges valid for edge slide*/ | 
					
						
							|  |  |  | #define DOUBLEOPFILL 16
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* calculates offset for co, based on fractal, sphere or smooth settings  */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | static void alter_co(float *co, EditEdge *edge, float smooth, float fractal, int beauty, float perc) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec1[3], fac; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(beauty & B_SMOOTH) { | 
					
						
							|  |  |  | 		/* we calculate an offset vector vec1[], to be added to *co */ | 
					
						
							|  |  |  | 		float len, fac, nor[3], nor1[3], nor2[3]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		VecSubf(nor, edge->v1->co, edge->v2->co); | 
					
						
							|  |  |  | 		len= 0.5f*Normalize(nor); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		VECCOPY(nor1, edge->v1->no); | 
					
						
							|  |  |  | 		VECCOPY(nor2, edge->v2->no); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/* cosine angle */ | 
					
						
							|  |  |  | 		fac= nor[0]*nor1[0] + nor[1]*nor1[1] + nor[2]*nor1[2] ; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		vec1[0]= fac*nor1[0]; | 
					
						
							|  |  |  | 		vec1[1]= fac*nor1[1]; | 
					
						
							|  |  |  | 		vec1[2]= fac*nor1[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/* cosine angle */ | 
					
						
							|  |  |  | 		fac= -nor[0]*nor2[0] - nor[1]*nor2[1] - nor[2]*nor2[2] ; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		vec1[0]+= fac*nor2[0]; | 
					
						
							|  |  |  | 		vec1[1]+= fac*nor2[1]; | 
					
						
							|  |  |  | 		vec1[2]+= fac*nor2[2]; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* falloff for multi subdivide */ | 
					
						
							|  |  |  | 		smooth *= sqrt(fabs(1.0f - 2.0f*fabs(0.5f-perc))); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		vec1[0]*= smooth*len; | 
					
						
							|  |  |  | 		vec1[1]*= smooth*len; | 
					
						
							|  |  |  | 		vec1[2]*= smooth*len; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		co[0] += vec1[0]; | 
					
						
							|  |  |  | 		co[1] += vec1[1]; | 
					
						
							|  |  |  | 		co[2] += vec1[2]; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	else if(beauty & B_SPHERE) { /* subdivide sphere */ | 
					
						
							|  |  |  | 		Normalize(co); | 
					
						
							|  |  |  | 		co[0]*= smooth; | 
					
						
							|  |  |  | 		co[1]*= smooth; | 
					
						
							|  |  |  | 		co[2]*= smooth; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	if(beauty & B_FRACTAL) { | 
					
						
							|  |  |  | 		fac= fractal*VecLenf(edge->v1->co, edge->v2->co); | 
					
						
							|  |  |  | 		vec1[0]= fac*(float)(0.5-BLI_drand()); | 
					
						
							|  |  |  | 		vec1[1]= fac*(float)(0.5-BLI_drand()); | 
					
						
							|  |  |  | 		vec1[2]= fac*(float)(0.5-BLI_drand()); | 
					
						
							|  |  |  | 		VecAddf(co, co, vec1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* assumes in the edge is the correct interpolated vertices already */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /* percent defines the interpolation, smooth, fractal and beauty are for special options */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* results in new vertex with correct coordinate, vertex normal and weight group info */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | static EditVert *subdivide_edge_addvert(EditMesh *em, EditEdge *edge, float smooth, float fractal, int beauty, float percent) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert *ev; | 
					
						
							|  |  |  | 	float co[3]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	co[0] = (edge->v2->co[0]-edge->v1->co[0])*percent + edge->v1->co[0]; | 
					
						
							|  |  |  | 	co[1] = (edge->v2->co[1]-edge->v1->co[1])*percent + edge->v1->co[1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	co[2] = (edge->v2->co[2]-edge->v1->co[2])*percent + edge->v1->co[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* offset for smooth or sphere or fractal */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	alter_co(co, edge, smooth, fractal, beauty, percent); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* clip if needed by mirror modifier */ | 
					
						
							|  |  |  | 	if (edge->v1->f2) { | 
					
						
							|  |  |  | 		if ( edge->v1->f2 & edge->v2->f2 & 1) { | 
					
						
							|  |  |  | 			co[0]= 0.0f; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if ( edge->v1->f2 & edge->v2->f2 & 2) { | 
					
						
							|  |  |  | 			co[1]= 0.0f; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if ( edge->v1->f2 & edge->v2->f2 & 4) { | 
					
						
							|  |  |  | 			co[2]= 0.0f; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	ev = addvertlist(em, co, NULL); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* vert data (vgroups, ..) */ | 
					
						
							|  |  |  | 	EM_data_interp_from_verts(em, edge->v1, edge->v2, ev, percent); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* normal */ | 
					
						
							|  |  |  | 	ev->no[0] = (edge->v2->no[0]-edge->v1->no[0])*percent + edge->v1->no[0]; | 
					
						
							|  |  |  | 	ev->no[1] = (edge->v2->no[1]-edge->v1->no[1])*percent + edge->v1->no[1]; | 
					
						
							|  |  |  | 	ev->no[2] = (edge->v2->no[2]-edge->v1->no[2])*percent + edge->v1->no[2]; | 
					
						
							|  |  |  | 	Normalize(ev->no); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return ev; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void flipvertarray(EditVert** arr, short size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *hold; | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=0; i<size/2; i++) { | 
					
						
							|  |  |  | 		hold = arr[i]; | 
					
						
							|  |  |  | 		arr[i] = arr[size-i-1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		arr[size-i-1] = hold; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void facecopy(EditMesh *em, EditFace *source, EditFace *target) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float *v1 = source->v1->co, *v2 = source->v2->co, *v3 = source->v3->co; | 
					
						
							|  |  |  | 	float *v4 = source->v4? source->v4->co: NULL; | 
					
						
							|  |  |  | 	float w[4][4]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	CustomData_em_copy_data(&em->fdata, &em->fdata, source->data, &target->data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	target->mat_nr = source->mat_nr; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	target->flag   = source->flag; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	target->h	   = source->h; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	InterpWeightsQ3Dfl(v1, v2, v3, v4, target->v1->co, w[0]); | 
					
						
							|  |  |  | 	InterpWeightsQ3Dfl(v1, v2, v3, v4, target->v2->co, w[1]); | 
					
						
							|  |  |  | 	InterpWeightsQ3Dfl(v1, v2, v3, v4, target->v3->co, w[2]); | 
					
						
							|  |  |  | 	if (target->v4) | 
					
						
							|  |  |  | 		InterpWeightsQ3Dfl(v1, v2, v3, v4, target->v4->co, w[3]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	CustomData_em_interp(&em->fdata, &source->data, NULL, (float*)w, 1, target->data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_single(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts, int seltype) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge=NULL; | 
					
						
							|  |  |  | 	EditVert *v[4], **verts; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	short start=0, end, left, right, vertsize,i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT)	  { cedge = efa->e1; start = 0;} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	else if(efa->e2->f & SELECT) { cedge = efa->e2; start = 1;} | 
					
						
							|  |  |  | 	else if(efa->e3->f & SELECT) { cedge = efa->e3; start = 2;} | 
					
						
							|  |  |  | 	else if(efa->e4->f & SELECT) { cedge = efa->e4; start = 3;} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts to the array of new verts for cedge
 | 
					
						
							|  |  |  | 	verts = BLI_ghash_lookup(gh, cedge); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0] != v[start]) {flipvertarray(verts,numcuts+2);} | 
					
						
							|  |  |  | 	end	= (start+1)%4; | 
					
						
							|  |  |  | 	left   = (start+2)%4; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	right  = (start+3)%4; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			  end		 start | 
					
						
							|  |  |  | 			   3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   |		   | | 
					
						
							|  |  |  | 			   |		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |		   | | 
					
						
							|  |  |  | 			   ------------- | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			  left	   right | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	where start,end,left, right are indexes of EditFace->v1, etc (stored in v) | 
					
						
							|  |  |  | 	and 0,1,2... are the indexes of the new verts stored in verts | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this depending on even or odd cuts | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |---*---*---|		  |---*---| | 
					
						
							|  |  |  | 			   |  /	 \  |		  |  / \  | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   | /	   \ |		  | /   \ | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |/		 \|		  |/	 \| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   -------------		  --------- | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make center face
 | 
					
						
							|  |  |  | 	if(vertsize % 2 == 0) { | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[(vertsize-1)/2],verts[((vertsize-1)/2)+1],v[left],v[right], NULL,NULL); | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		hold->e4->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 	}else{ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[(vertsize-1)/2],v[left],v[right],NULL, NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	facecopy(em, efa,hold); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make side faces
 | 
					
						
							|  |  |  | 	for(i=0;i<(vertsize-1)/2;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[i],verts[i+1],v[right],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 		if(i+1 != (vertsize-1)/2) { | 
					
						
							|  |  |  |             if(seltype == SUBDIV_SELECT_INNER) { | 
					
						
							|  |  |  | 	 		   hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[vertsize-2-i],verts[vertsize-1-i],v[left],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 		if(i+1 != (vertsize-1)/2) { | 
					
						
							|  |  |  |             if(seltype == SUBDIV_SELECT_INNER) { | 
					
						
							|  |  |  | 		 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_tri_single(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts, int seltype) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge=NULL; | 
					
						
							|  |  |  | 	EditVert *v[3], **verts; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	short start=0, end, op, vertsize,i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT)	  { cedge = efa->e1; start = 0;} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	else if(efa->e2->f & SELECT) { cedge = efa->e2; start = 1;} | 
					
						
							|  |  |  | 	else if(efa->e3->f & SELECT) { cedge = efa->e3; start = 2;} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts to the array of new verts for cedge
 | 
					
						
							|  |  |  | 	verts = BLI_ghash_lookup(gh, cedge); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0] != v[start]) {flipvertarray(verts,numcuts+2);} | 
					
						
							|  |  |  | 	   end	= (start+1)%3; | 
					
						
							|  |  |  | 	   op	 = (start+2)%3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			  end		 start | 
					
						
							|  |  |  | 			   3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   \		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				 \		 | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				   \	   | | 
					
						
							|  |  |  | 					 \	 | | 
					
						
							|  |  |  | 					   \   | | 
					
						
							|  |  |  | 						 \ | | 
					
						
							|  |  |  | 						   |op | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	where start,end,op are indexes of EditFace->v1, etc (stored in v) | 
					
						
							|  |  |  | 	and 0,1,2... are the indexes of the new verts stored in verts | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this depending on even or odd cuts | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			   3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   \	\  \   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				 \	\ \  | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				   \   \ \ | | 
					
						
							|  |  |  | 					 \  \ \| | 
					
						
							|  |  |  | 					   \ \\| | 
					
						
							|  |  |  | 						 \ | | 
					
						
							|  |  |  | 						   |op | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make side faces
 | 
					
						
							|  |  |  | 	for(i=0;i<(vertsize-1);i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[i],verts[i+1],v[op],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(i+1 != vertsize-1) { | 
					
						
							|  |  |  |             if(seltype == SUBDIV_SELECT_INNER) { | 
					
						
							|  |  |  | 		 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_double_op(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge[2]={NULL, NULL}; | 
					
						
							|  |  |  | 	EditVert *v[4], **verts[2]; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	short start=0, end, left, right, vertsize,i; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT)	  { cedge[0] = efa->e1;  cedge[1] = efa->e3; start = 0;} | 
					
						
							|  |  |  | 	else if(efa->e2->f & SELECT)	  { cedge[0] = efa->e2;  cedge[1] = efa->e4; start = 1;} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, cedge[0]); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, cedge[1]); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != v[start]) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							|  |  |  | 	end	= (start+1)%4; | 
					
						
							|  |  |  | 	left   = (start+2)%4; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	right  = (start+3)%4; | 
					
						
							|  |  |  | 	if(verts[1][0] != v[left]) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			  end		 start | 
					
						
							|  |  |  | 			   3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   |		   | | 
					
						
							|  |  |  | 			   |		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |		   | | 
					
						
							|  |  |  | 			   |---*---*---| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   0   1   2   3 | 
					
						
							|  |  |  | 			  left	   right | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this depending on even or odd cuts | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   |   |   |   | | 
					
						
							|  |  |  | 			   |   |   |   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |   |   |   | | 
					
						
							|  |  |  | 			   |---*---*---| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Make side faces
 | 
					
						
							|  |  |  | 	for(i=0;i<vertsize-1;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[0][i],verts[0][i+1],verts[1][vertsize-2-i],verts[1][vertsize-1-i],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(i < vertsize-2) { | 
					
						
							|  |  |  | 			hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 			hold->e2->f2 |= DOUBLEOPFILL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_double_adj_path(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge[2]={NULL, NULL}; | 
					
						
							|  |  |  | 	EditVert *v[4], **verts[2]; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	short start=0, start2=0, vertsize,i; | 
					
						
							|  |  |  | 	int ctrl= 0; // XXX
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT && efa->e2->f & SELECT) {cedge[0] = efa->e1;  cedge[1] = efa->e2; start = 0; start2 = 1;} | 
					
						
							|  |  |  | 	if(efa->e2->f & SELECT && efa->e3->f & SELECT) {cedge[0] = efa->e2;  cedge[1] = efa->e3; start = 1; start2 = 2;} | 
					
						
							|  |  |  | 	if(efa->e3->f & SELECT && efa->e4->f & SELECT) {cedge[0] = efa->e3;  cedge[1] = efa->e4; start = 2; start2 = 3;} | 
					
						
							|  |  |  | 	if(efa->e4->f & SELECT && efa->e1->f & SELECT) {cedge[0] = efa->e4;  cedge[1] = efa->e1; start = 3; start2 = 0;} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, cedge[0]); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, cedge[1]); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != v[start]) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != v[start2]) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   end		 start | 
					
						
							|  |  |  | 				3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		start2 0|---*---*---| | 
					
						
							|  |  |  | 				|		   | | 
					
						
							|  |  |  | 			   1*		   | | 
					
						
							|  |  |  | 				|		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   2*		   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		 end2  3|-----------| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this depending on even or odd cuts | 
					
						
							|  |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   | /   /   / | | 
					
						
							|  |  |  | 			   *   /   /   | | 
					
						
							|  |  |  | 			   | /   /	 | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   *   /	   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   | /		 | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |-----------| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make outside tris
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][vertsize-2],verts[0][vertsize-1],verts[1][1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* when ctrl is depressed, only want verts on the cutline selected */ | 
					
						
							|  |  |  | 	if (ctrl) | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][0],verts[1][vertsize-1],v[(start2+2)%4],NULL,NULL,NULL); | 
					
						
							|  |  |  | 	/* when ctrl is depressed, only want verts on the cutline selected */ | 
					
						
							|  |  |  | 	if (ctrl) | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	//if(scene->toolsettings->editbutflag & B_AUTOFGON) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//	hold->e1->h |= EM_FGON;
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	//}
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Make side faces
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[0][i],verts[0][i+1],verts[1][vertsize-1-(i+1)],verts[1][vertsize-1-i],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//EM_fgon_flags(em);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_double_adj_fan(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge[2]={NULL, NULL}; | 
					
						
							|  |  |  | 	EditVert *v[4], *op=NULL, **verts[2]; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	short start=0, start2=0, vertsize,i; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT && efa->e2->f & SELECT) {cedge[0] = efa->e1;  cedge[1] = efa->e2; start = 0; start2 = 1; op = efa->v4;} | 
					
						
							|  |  |  | 	if(efa->e2->f & SELECT && efa->e3->f & SELECT) {cedge[0] = efa->e2;  cedge[1] = efa->e3; start = 1; start2 = 2; op = efa->v1;} | 
					
						
							|  |  |  | 	if(efa->e3->f & SELECT && efa->e4->f & SELECT) {cedge[0] = efa->e3;  cedge[1] = efa->e4; start = 2; start2 = 3; op = efa->v2;} | 
					
						
							|  |  |  | 	if(efa->e4->f & SELECT && efa->e1->f & SELECT) {cedge[0] = efa->e4;  cedge[1] = efa->e1; start = 3; start2 = 0; op = efa->v3;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, cedge[0]); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, cedge[1]); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != v[start]) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != v[start2]) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   end		 start | 
					
						
							|  |  |  | 				3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		start2 0|---*---*---| | 
					
						
							|  |  |  | 				|		   | | 
					
						
							|  |  |  | 			   1*		   | | 
					
						
							|  |  |  | 				|		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   2*		   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		 end2  3|-----------|op | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this (warning horrible ascii art follows) | 
					
						
							|  |  |  | 			   |---*---*---| | 
					
						
							|  |  |  | 			   | \  \   \  | | 
					
						
							|  |  |  | 			   *---\  \  \ | | 
					
						
							|  |  |  | 			   |   \ \ \  \| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   *---- \ \  \ | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |    ---  \\\| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |-----------| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i=0;i<=numcuts;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, op,verts[1][numcuts-i],verts[1][numcuts-i+1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, op,verts[0][i],verts[0][i+1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_double_adj_inner(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge[2]={NULL, NULL}; | 
					
						
							|  |  |  | 	EditVert *v[4], *op=NULL, **verts[2],**inner; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	short start=0, start2=0, vertsize,i; | 
					
						
							|  |  |  | 	float co[3]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT && efa->e2->f & SELECT) {cedge[0] = efa->e1;  cedge[1] = efa->e2; start = 0; start2 = 1; op = efa->v4;} | 
					
						
							|  |  |  | 	if(efa->e2->f & SELECT && efa->e3->f & SELECT) {cedge[0] = efa->e2;  cedge[1] = efa->e3; start = 1; start2 = 2; op = efa->v1;} | 
					
						
							|  |  |  | 	if(efa->e3->f & SELECT && efa->e4->f & SELECT) {cedge[0] = efa->e3;  cedge[1] = efa->e4; start = 2; start2 = 3; op = efa->v2;} | 
					
						
							|  |  |  | 	if(efa->e4->f & SELECT && efa->e1->f & SELECT) {cedge[0] = efa->e4;  cedge[1] = efa->e1; start = 3; start2 = 0; op = efa->v3;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, cedge[0]); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, cedge[1]); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != v[start]) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != v[start2]) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   end		 start | 
					
						
							|  |  |  | 				3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		start2 0|---*---*---| | 
					
						
							|  |  |  | 				|		   | | 
					
						
							|  |  |  | 			   1*		   | | 
					
						
							|  |  |  | 				|		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   2*		   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		 end2  3|-----------|op | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this (warning horrible ascii art follows) | 
					
						
							|  |  |  | 			   |---*-----*---| | 
					
						
							|  |  |  | 			   | *     /     | | 
					
						
							|  |  |  | 			   *   \ /       | | 
					
						
							|  |  |  | 			   |    *        | | 
					
						
							|  |  |  | 			   | /	  \	     | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   *        \    | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |           \ | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |-------------| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Add Inner Vert(s)
 | 
					
						
							|  |  |  | 	inner = MEM_mallocN(sizeof(EditVert*)*numcuts,"New inner verts"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=0;i<numcuts;i++) { | 
					
						
							|  |  |  | 		co[0] = (verts[0][numcuts-i]->co[0] + verts[1][i+1]->co[0] ) / 2 ; | 
					
						
							|  |  |  | 		co[1] = (verts[0][numcuts-i]->co[1] + verts[1][i+1]->co[1] ) / 2 ; | 
					
						
							|  |  |  | 		co[2] = (verts[0][numcuts-i]->co[2] + verts[1][i+1]->co[2] ) / 2 ; | 
					
						
							|  |  |  | 		inner[i] = addvertlist(em, co, NULL); | 
					
						
							|  |  |  | 		inner[i]->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		EM_data_interp_from_verts(em, verts[0][numcuts-i], verts[1][i+1], inner[i], 0.5f); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Add Corner Quad
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][numcuts+1],verts[1][1],inner[0],verts[0][numcuts],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 	hold->e3->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Add Bottom Quads
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][0],verts[0][1],inner[numcuts-1],op,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e2->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, op,inner[numcuts-1],verts[1][numcuts],verts[1][numcuts+1],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e2->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	//if(scene->toolsettings->editbutflag & B_AUTOFGON) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//	hold->e1->h |= EM_FGON;
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	//}
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Add Fill Quads (if # cuts > 1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts-1;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, inner[i],verts[1][i+1],verts[1][i+2],inner[i+1],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, inner[i],inner[i+1],verts[0][numcuts-1-i],verts[0][numcuts-i],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		hold->e4->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 		//if(scene->toolsettings->editbutflag & B_AUTOFGON) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		//	hold->e1->h |= EM_FGON;
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		//}
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//EM_fgon_flags(em);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(inner); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_tri_double(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge[2]={NULL, NULL}; | 
					
						
							|  |  |  | 	EditVert *v[3], **verts[2]; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	short start=0, start2=0, vertsize,i; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT && efa->e2->f & SELECT) {cedge[0] = efa->e1;  cedge[1] = efa->e2; start = 0; start2 = 1;} | 
					
						
							|  |  |  | 	if(efa->e2->f & SELECT && efa->e3->f & SELECT) {cedge[0] = efa->e2;  cedge[1] = efa->e3; start = 1; start2 = 2;} | 
					
						
							|  |  |  | 	if(efa->e3->f & SELECT && efa->e1->f & SELECT) {cedge[0] = efa->e3;  cedge[1] = efa->e1; start = 2; start2 = 0;} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, cedge[0]); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, cedge[1]); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != v[start]) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != v[start2]) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   end		 start | 
					
						
							|  |  |  | 				3   2   1   0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		start2 0|---*---*---| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				|		 / | 
					
						
							|  |  |  | 			   1*	   / | 
					
						
							|  |  |  | 				|	 / | 
					
						
							|  |  |  | 			   2*   / | 
					
						
							|  |  |  | 				| / | 
					
						
							|  |  |  | 		 end2  3| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	We will fill this case like this or this depending on even or odd cuts | 
					
						
							|  |  |  | 			   |---*---*---| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   | /   /   / | 
					
						
							|  |  |  | 			   *   /   / | 
					
						
							|  |  |  | 			   | /   / | 
					
						
							|  |  |  | 			   *   / | 
					
						
							|  |  |  | 			   | / | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   | | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make outside tri
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][vertsize-2],verts[0][vertsize-1],verts[1][1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e3->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Make side faces
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[0][i],verts[0][i+1],verts[1][vertsize-1-(i+1)],verts[1][vertsize-1-i],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_triple(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge[3]={0}; | 
					
						
							|  |  |  | 	EditVert *v[4], **verts[3]; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	short start=0, start2=0, start3=0, vertsize, i, repeats; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!(efa->e1->f & SELECT)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		cedge[0] = efa->e2; | 
					
						
							|  |  |  | 		cedge[1] = efa->e3; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		cedge[2] = efa->e4; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		start = 1;start2 = 2;start3 = 3; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if(!(efa->e2->f & SELECT)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		cedge[0] = efa->e3; | 
					
						
							|  |  |  | 		cedge[1] = efa->e4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		cedge[2] = efa->e1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		start = 2;start2 = 3;start3 = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if(!(efa->e3->f & SELECT)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		cedge[0] = efa->e4; | 
					
						
							|  |  |  | 		cedge[1] = efa->e1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		cedge[2] = efa->e2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		start = 3;start2 = 0;start3 = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if(!(efa->e4->f & SELECT)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		cedge[0] = efa->e1; | 
					
						
							|  |  |  | 		cedge[1] = efa->e2; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		cedge[2] = efa->e3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		start = 0;start2 = 1;start3 = 2; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, cedge[0]); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, cedge[1]); | 
					
						
							|  |  |  | 	verts[2] = BLI_ghash_lookup(gh, cedge[2]); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(verts[0][0] != v[start]) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != v[start2]) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							|  |  |  | 	if(verts[2][0] != v[start3]) {flipvertarray(verts[2],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 We should have something like this now | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	 start2 | 
					
						
							|  |  |  | 	 3   2   1   0 | 
					
						
							|  |  |  | 	 start3 0|---*---*---|3 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |		   | | 
					
						
							|  |  |  | 	 1*		   *2 | 
					
						
							|  |  |  | 	 |		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 2*		   *1 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 3|-----------|0 start | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	 We will fill this case like this or this depending on even or odd cuts | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 there are a couple of differences. For odd cuts, there is a tri in the | 
					
						
							|  |  |  | 	 middle as well as 1 quad at the bottom (not including the extra quads | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 for odd cuts > 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 For even cuts, there is a quad in the middle and 2 quads on the bottom | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 they are numbered here for clarity | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 1 outer tris and bottom quads | 
					
						
							|  |  |  | 	 2 inner tri or quad | 
					
						
							|  |  |  | 	 3 repeating quads | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |---*---*---*---| | 
					
						
							|  |  |  | 	 |1/   /  \   \ 1| | 
					
						
							|  |  |  | 	 |/ 3 /	\  3 \| | 
					
						
							|  |  |  | 	 *  /	2   \   * | 
					
						
							|  |  |  | 	 | /		  \  | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |/			\ | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 *---------------* | 
					
						
							|  |  |  | 	 |	  3		| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |			   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 *---------------* | 
					
						
							|  |  |  | 	 |			   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |	  1		| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |			   | | 
					
						
							|  |  |  | 	 |---------------| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |---*---*---*---*---| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 | 1/   /	 \   \ 1| | 
					
						
							|  |  |  | 	 | /   /	   \   \ | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |/ 3 /		 \ 3 \| | 
					
						
							|  |  |  | 	 *   /		   \   * | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |  /			 \  | | 
					
						
							|  |  |  | 	 | /	   2	   \ | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |/				 \| | 
					
						
							|  |  |  | 	 *-------------------* | 
					
						
							|  |  |  | 	 |				   | | 
					
						
							|  |  |  | 	 |		 3		 | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |				   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 *-------------------* | 
					
						
							|  |  |  | 	 |				   | | 
					
						
							|  |  |  | 	 |		 1		 | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |				   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 *-------------------* | 
					
						
							|  |  |  | 	 |				   | | 
					
						
							|  |  |  | 	 |		1		  | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	 |				   | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 |-------------------| | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make outside tris
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][vertsize-2],verts[0][vertsize-1],verts[1][1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e3->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							|  |  |  | 	hold = addfacelist(em, verts[1][vertsize-2],verts[1][vertsize-1],verts[2][1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e3->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Make bottom quad
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	hold = addfacelist(em, verts[0][0],verts[0][1],verts[2][vertsize-2],verts[2][vertsize-1],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	hold->e2->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//If it is even cuts, add the 2nd lower quad
 | 
					
						
							|  |  |  | 	if(numcuts % 2 == 0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[0][1],verts[0][2],verts[2][vertsize-3],verts[2][vertsize-2],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		// Also Make inner quad
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[1][numcuts/2],verts[1][(numcuts/2)+1],verts[2][numcuts/2],verts[0][(numcuts/2)+1],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 		//if(scene->toolsettings->editbutflag & B_AUTOFGON) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		//	hold->e3->h |= EM_FGON;
 | 
					
						
							|  |  |  | 		//}
 | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 		repeats = (numcuts / 2) -1; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		// Make inner tri
 | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[1][(numcuts/2)+1],verts[2][(numcuts/2)+1],verts[0][(numcuts/2)+1],NULL,NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 		//if(scene->toolsettings->editbutflag & B_AUTOFGON) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		//	hold->e2->h |= EM_FGON;
 | 
					
						
							|  |  |  | 		//}
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		repeats = ((numcuts+1) / 2)-1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// cuts for 1 and 2 do not have the repeating quads
 | 
					
						
							|  |  |  | 	if(numcuts < 3) {repeats = 0;} | 
					
						
							|  |  |  | 	for(i=0;i<repeats;i++) { | 
					
						
							|  |  |  | 		//Make side repeating Quads
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		hold = addfacelist(em, verts[1][i+1],verts[1][i+2],verts[0][vertsize-i-3],verts[0][vertsize-i-2],NULL,NULL); | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[1][vertsize-i-3],verts[1][vertsize-i-2],verts[2][i+1],verts[2][i+2],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold->e4->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	// Do repeating bottom quads
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=0;i<repeats;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		if(numcuts % 2 == 1) { | 
					
						
							|  |  |  | 			hold = addfacelist(em, verts[0][1+i],verts[0][2+i],verts[2][vertsize-3-i],verts[2][vertsize-2-i],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			hold = addfacelist(em, verts[0][2+i],verts[0][3+i],verts[2][vertsize-4-i],verts[2][vertsize-3-i],NULL,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		facecopy(em, efa,hold); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//EM_fgon_flags(em);
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | static void fill_quad_quadruple(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts, float smooth, float fractal, int beauty) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert **verts[4], ***innerverts; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	EditFace *hold; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditEdge temp; | 
					
						
							|  |  |  | 	short vertsize, i, j; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, efa->e1); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, efa->e2); | 
					
						
							|  |  |  | 	verts[2] = BLI_ghash_lookup(gh, efa->e3); | 
					
						
							|  |  |  | 	verts[3] = BLI_ghash_lookup(gh, efa->e4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != efa->v1) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != efa->v2) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(verts[2][0] == efa->v3) {flipvertarray(verts[2],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[3][0] == efa->v4) {flipvertarray(verts[3],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 					  1 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				3   2   1   0 | 
					
						
							|  |  |  | 			   0|---*---*---|0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|           | | 
					
						
							|  |  |  | 			   1*           *1 | 
					
						
							|  |  |  | 		     2  |           |   4 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   2*           *2 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|           | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   3|---*---*---|3 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				3   2   1   0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					  3 | 
					
						
							|  |  |  | 	// we will fill a 2 dim array of editvert*s to make filling easier
 | 
					
						
							|  |  |  | 	//  the innervert order is shown
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				0   0---1---2---3 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					|   |   |   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				1   0---1---2---3 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					|   |   |   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				2   0---1---2---3 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					|   |   |   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				3   0---1---2---3 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	innerverts = MEM_mallocN(sizeof(EditVert*)*(numcuts+2),"quad-quad subdiv inner verts outer array"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=0;i<numcuts+2;i++) { | 
					
						
							|  |  |  | 		innerverts[i] = MEM_mallocN(sizeof(EditVert*)*(numcuts+2),"quad-quad subdiv inner verts inner array"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// first row is e1 last row is e3
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts+2;i++) { | 
					
						
							|  |  |  | 		innerverts[0][i]		  = verts[0][(numcuts+1)-i]; | 
					
						
							|  |  |  | 		innerverts[numcuts+1][i]  = verts[2][(numcuts+1)-i]; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=1;i<=numcuts;i++) { | 
					
						
							|  |  |  | 		/* we create a fake edge for the next loop */ | 
					
						
							|  |  |  | 		temp.v2 = innerverts[i][0]			= verts[1][i]; | 
					
						
							|  |  |  | 		temp.v1 = innerverts[i][numcuts+1]  = verts[3][i]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		for(j=1;j<=numcuts;j++) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			float percent= (float)j/(float)(numcuts+1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			innerverts[i][(numcuts+1)-j]= subdivide_edge_addvert(em, &temp, smooth, fractal, beauty, percent); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Fill with faces
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts+1;i++) { | 
					
						
							|  |  |  | 		for(j=0;j<numcuts+1;j++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			hold = addfacelist(em, innerverts[i][j+1],innerverts[i][j],innerverts[i+1][j],innerverts[i+1][j+1],NULL,NULL); | 
					
						
							|  |  |  | 			hold->e1->f2 = EDGENEW; | 
					
						
							|  |  |  | 			hold->e2->f2 = EDGENEW; | 
					
						
							|  |  |  | 			hold->e3->f2 = EDGENEW; | 
					
						
							|  |  |  | 			hold->e4->f2 = EDGENEW; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(i != 0) { hold->e1->f2 |= EDGEINNER; } | 
					
						
							|  |  |  | 			if(j != 0) { hold->e2->f2 |= EDGEINNER; } | 
					
						
							|  |  |  | 			if(i != numcuts) { hold->e3->f2 |= EDGEINNER; } | 
					
						
							|  |  |  | 			if(j != numcuts) { hold->e4->f2 |= EDGEINNER; } | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			facecopy(em, efa,hold); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	// Clean up our dynamic multi-dim array
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts+2;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	   MEM_freeN(innerverts[i]); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MEM_freeN(innerverts); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | static void fill_tri_triple(EditMesh *em, EditFace *efa, struct GHash *gh, int numcuts, float smooth, float fractal, int beauty) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert **verts[3], ***innerverts; | 
					
						
							|  |  |  | 	short vertsize, i, j; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	EditFace *hold; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditEdge temp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts[0] and [1] to the array of new verts for cedge[0] and cedge[1]
 | 
					
						
							|  |  |  | 	verts[0] = BLI_ghash_lookup(gh, efa->e1); | 
					
						
							|  |  |  | 	verts[1] = BLI_ghash_lookup(gh, efa->e2); | 
					
						
							|  |  |  | 	verts[2] = BLI_ghash_lookup(gh, efa->e3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = numcuts+2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0][0] != efa->v1) {flipvertarray(verts[0],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(verts[1][0] != efa->v2) {flipvertarray(verts[1],numcuts+2);} | 
					
						
							|  |  |  | 	if(verts[2][0] != efa->v3) {flipvertarray(verts[2],numcuts+2);} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 					   3 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				3   2   1   0 | 
					
						
							|  |  |  | 			   0|---*---*---|3 | 
					
						
							|  |  |  | 				|		  / | 
					
						
							|  |  |  | 		  1	1*		*2 | 
					
						
							|  |  |  | 				|	  / | 
					
						
							|  |  |  | 			   2*	*1	   2 | 
					
						
							|  |  |  | 				|  / | 
					
						
							|  |  |  | 			   3|/ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	we will fill a 2 dim array of editvert*s to make filling easier | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						3 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			 0  0---1---2---3---4 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				| / | /  |/  | / | 
					
						
							|  |  |  | 			 1  0---1----2---3 | 
					
						
							|  |  |  | 	   1		| /  | / | / | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			 2  0----1---2	 2 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				|  / |  / | 
					
						
							|  |  |  | 				|/   |/ | 
					
						
							|  |  |  | 			 3  0---1 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|  / | 
					
						
							|  |  |  | 				|/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			 4  0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	innerverts = MEM_mallocN(sizeof(EditVert*)*(numcuts+2),"tri-tri subdiv inner verts outer array"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=0;i<numcuts+2;i++) { | 
					
						
							|  |  |  | 		  innerverts[i] = MEM_mallocN(sizeof(EditVert*)*((numcuts+2)-i),"tri-tri subdiv inner verts inner array"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//top row is e3 backwards
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts+2;i++) { | 
					
						
							|  |  |  | 		  innerverts[0][i]		  = verts[2][(numcuts+1)-i]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(i=1;i<=numcuts+1;i++) { | 
					
						
							|  |  |  | 		//fake edge, first vert is from e1, last is from e2
 | 
					
						
							|  |  |  | 		temp.v1= innerverts[i][0]			  = verts[0][i]; | 
					
						
							|  |  |  | 		temp.v2= innerverts[i][(numcuts+1)-i]  = verts[1][(numcuts+1)-i]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(j=1;j<(numcuts+1)-i;j++) { | 
					
						
							|  |  |  | 			float percent= (float)j/(float)((numcuts+1)-i); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			innerverts[i][((numcuts+1)-i)-j]= subdivide_edge_addvert(em, &temp, smooth, fractal, beauty, 1-percent); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Now fill the verts with happy little tris :)
 | 
					
						
							|  |  |  | 	for(i=0;i<=numcuts+1;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		for(j=0;j<(numcuts+1)-i;j++) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			//We always do the first tri
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			hold = addfacelist(em, innerverts[i][j+1],innerverts[i][j],innerverts[i+1][j],NULL,NULL,NULL); | 
					
						
							|  |  |  | 			hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 			hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 			hold->e3->f2 |= EDGENEW; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(i != 0) { hold->e1->f2 |= EDGEINNER; } | 
					
						
							|  |  |  | 			if(j != 0) { hold->e2->f2 |= EDGEINNER; } | 
					
						
							|  |  |  | 			if(j+1 != (numcuts+1)-i) {hold->e3->f2 |= EDGEINNER;} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			facecopy(em, efa,hold); | 
					
						
							|  |  |  | 			//if there are more to come, we do the 2nd
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(j+1 <= numcuts-i) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				hold = addfacelist(em, innerverts[i+1][j],innerverts[i+1][j+1],innerverts[i][j+1],NULL,NULL,NULL); | 
					
						
							|  |  |  | 				facecopy(em, efa,hold); | 
					
						
							|  |  |  | 				hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 				hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 				hold->e3->f2 |= EDGENEW; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Clean up our dynamic multi-dim array
 | 
					
						
							|  |  |  | 	for(i=0;i<numcuts+2;i++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		MEM_freeN(innerverts[i]); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MEM_freeN(innerverts); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //Next two fill types are for knife exact only and are provided to allow for knifing through vertices
 | 
					
						
							|  |  |  | //This means there is no multicut!
 | 
					
						
							|  |  |  | static void fill_quad_doublevert(EditMesh *em, EditFace *efa, int v1, int v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 		Depending on which two vertices have been knifed through (v1 and v2), we | 
					
						
							|  |  |  | 		triangulate like the patterns below. | 
					
						
							|  |  |  | 				X-------|	|-------X | 
					
						
							|  |  |  | 				| \  	|	|     / | | 
					
						
							|  |  |  | 				|   \	|	|   /	| | 
					
						
							|  |  |  | 				|	  \	|	| /	    | | 
					
						
							|  |  |  | 				--------X	X-------- | 
					
						
							|  |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(v1 == 1 && v2 == 3){ | 
					
						
							|  |  |  | 		hold= addfacelist(em, efa->v1, efa->v2, efa->v3, 0, efa, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold= addfacelist(em, efa->v1, efa->v3, efa->v4, 0, efa, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else{ | 
					
						
							|  |  |  | 		hold= addfacelist(em, efa->v1, efa->v2, efa->v4, 0, efa, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		hold= addfacelist(em, efa->v2, efa->v3, efa->v4, 0, efa, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fill_quad_singlevert(EditMesh *em, EditFace *efa, struct GHash *gh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *cedge=NULL; | 
					
						
							|  |  |  | 	EditVert *v[4], **verts; | 
					
						
							|  |  |  | 	EditFace *hold; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	short start=0, end, left, right, vertsize; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	v[0] = efa->v1; | 
					
						
							|  |  |  | 	v[1] = efa->v2; | 
					
						
							|  |  |  | 	v[2] = efa->v3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	v[3] = efa->v4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(efa->e1->f & SELECT)	  { cedge = efa->e1; start = 0;} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	else if(efa->e2->f & SELECT) { cedge = efa->e2; start = 1;} | 
					
						
							|  |  |  | 	else if(efa->e3->f & SELECT) { cedge = efa->e3; start = 2;} | 
					
						
							|  |  |  | 	else if(efa->e4->f & SELECT) { cedge = efa->e4; start = 3;} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Point verts to the array of new verts for cedge
 | 
					
						
							|  |  |  | 	verts = BLI_ghash_lookup(gh, cedge); | 
					
						
							|  |  |  | 	//This is the index size of the verts array
 | 
					
						
							|  |  |  | 	vertsize = 3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Is the original v1 the same as the first vert on the selected edge?
 | 
					
						
							|  |  |  | 	// if not, the edge is running the opposite direction in this face so flip
 | 
					
						
							|  |  |  | 	// the array to the correct direction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(verts[0] != v[start]) {flipvertarray(verts,3);} | 
					
						
							|  |  |  | 	end	= (start+1)%4; | 
					
						
							|  |  |  | 	left   = (start+2)%4; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	right  = (start+3)%4; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | 	We should have something like this now | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			  end		 start | 
					
						
							|  |  |  | 			   2     1     0 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   |-----*-----| | 
					
						
							|  |  |  | 			   |		   | | 
					
						
							|  |  |  | 			   |		   | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			   |		   | | 
					
						
							|  |  |  | 			   ------------- | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			  left	   right | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	where start,end,left, right are indexes of EditFace->v1, etc (stored in v) | 
					
						
							|  |  |  | 	and 0,1,2 are the indexes of the new verts stored in verts. We fill like | 
					
						
							|  |  |  | 	this, depending on whether its vertex 'left' or vertex 'right' thats | 
					
						
							|  |  |  | 	been knifed through... | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				|---*---|	|---*---| | 
					
						
							|  |  |  | 				|  /	|	|    \  | | 
					
						
							|  |  |  | 				| /		|	|	  \ | | 
					
						
							|  |  |  | 				|/		|	|	   \| | 
					
						
							|  |  |  | 				X--------	--------X | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(v[left]->f1){ | 
					
						
							|  |  |  | 		//triangle is composed of cutvert, end and left
 | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[1],v[end],v[left],NULL, NULL,NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		//quad is composed of cutvert, left, right and start
 | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[1],v[left],v[right],v[start], NULL, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e4->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(v[right]->f1){ | 
					
						
							|  |  |  | 		//triangle is composed of cutvert, right and start
 | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[1],v[right],v[start], NULL, NULL, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							|  |  |  | 		//quad is composed of cutvert, end, left, right
 | 
					
						
							|  |  |  | 		hold = addfacelist(em, verts[1],v[end], v[left], v[right], NULL, NULL); | 
					
						
							|  |  |  | 		hold->e1->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e2->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e3->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e4->f2 |= EDGENEW; | 
					
						
							|  |  |  | 		hold->e4->f2 |= EDGEINNER; | 
					
						
							|  |  |  | 		facecopy(em, efa, hold); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This function takes an example edge, the current point to create and
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | // the total # of points to create, then creates the point and return the
 | 
					
						
							|  |  |  | // editvert pointer to it.
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | static EditVert *subdivideedgenum(EditMesh *em, EditEdge *edge, int curpoint, int totpoint, float smooth, float fractal, int beauty) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert *ev; | 
					
						
							|  |  |  | 	float percent; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (beauty & (B_PERCENTSUBD) && totpoint == 1) | 
					
						
							|  |  |  | 		//percent=(float)(edge->tmp.l)/32768.0f;
 | 
					
						
							|  |  |  | 		percent= edge->tmp.fp; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		percent= (float)curpoint/(float)(totpoint+1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ev= subdivide_edge_addvert(em, edge, smooth, fractal, beauty, percent); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	ev->f = edge->v1->f; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return ev; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | void esubdivideflag(Object *obedit, EditMesh *em, int flag, float smooth, float fractal, int beauty, int numcuts, int seltype) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditFace *ef; | 
					
						
							|  |  |  | 	EditEdge *eed, *cedge, *sort[4]; | 
					
						
							|  |  |  | 	EditVert *eve, **templist; | 
					
						
							|  |  |  | 	struct GHash *gh; | 
					
						
							|  |  |  | 	float length[4], v1mat[3], v2mat[3], v3mat[3], v4mat[3]; | 
					
						
							|  |  |  | 	int i, j, edgecount, touchcount, facetype,hold; | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	ModifierData *md= obedit->modifiers.first; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	int ctrl= 0; // XXX
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//Set faces f1 to 0 cause we need it later
 | 
					
						
							|  |  |  | 	for(ef=em->faces.first;ef;ef = ef->next) ef->f1 = 0; | 
					
						
							|  |  |  | 	for(eve=em->verts.first; eve; eve=eve->next) { | 
					
						
							|  |  |  | 		if(!(beauty & B_KNIFE)) /* knife sets this flag for vertex cuts */ | 
					
						
							|  |  |  | 			eve->f1 = 0; | 
					
						
							|  |  |  | 		eve->f2 = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (; md; md=md->next) { | 
					
						
							|  |  |  | 		if (md->type==eModifierType_Mirror) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			MirrorModifierData *mmd = (MirrorModifierData*) md; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(mmd->flag & MOD_MIR_CLIPPING) { | 
					
						
							|  |  |  | 				for (eve= em->verts.first; eve; eve= eve->next) { | 
					
						
							|  |  |  | 					eve->f2= 0; | 
					
						
							|  |  |  | 					switch(mmd->axis){ | 
					
						
							|  |  |  | 						case 0: | 
					
						
							|  |  |  | 							if (fabs(eve->co[0]) < mmd->tolerance) | 
					
						
							|  |  |  | 								eve->f2 |= 1; | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						case 1: | 
					
						
							|  |  |  | 							if (fabs(eve->co[1]) < mmd->tolerance) | 
					
						
							|  |  |  | 								eve->f2 |= 2; | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						case 2: | 
					
						
							|  |  |  | 							if (fabs(eve->co[2]) < mmd->tolerance) | 
					
						
							|  |  |  | 								eve->f2 |= 4; | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//Flush vertex flags upward to the edges
 | 
					
						
							|  |  |  | 	for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 		//if(eed->f & flag && eed->v1->f == eed->v2->f) {
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		//	eed->f |= eed->v1->f;
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		// }
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		eed->f2 = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eed->f & flag) { | 
					
						
							|  |  |  | 			eed->f2	|= EDGEOLD; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// We store an array of verts for each edge that is subdivided,
 | 
					
						
							|  |  |  | 	// we put this array as a value in a ghash which is keyed by the EditEdge*
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Now for beauty subdivide deselect edges based on length
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(beauty & B_BEAUTY) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(ef = em->faces.first;ef;ef = ef->next) { | 
					
						
							|  |  |  | 			if(!ef->v4) { | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(ef->f & SELECT) { | 
					
						
							|  |  |  | 				VECCOPY(v1mat, ef->v1->co); | 
					
						
							|  |  |  | 				VECCOPY(v2mat, ef->v2->co); | 
					
						
							|  |  |  | 				VECCOPY(v3mat, ef->v3->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				VECCOPY(v4mat, ef->v4->co); | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 				Mat4Mul3Vecfl(obedit->obmat, v1mat); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				Mat4Mul3Vecfl(obedit->obmat, v2mat); | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 				Mat4Mul3Vecfl(obedit->obmat, v3mat); | 
					
						
							|  |  |  | 				Mat4Mul3Vecfl(obedit->obmat, v4mat); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				length[0] = VecLenf(v1mat, v2mat); | 
					
						
							|  |  |  | 				length[1] = VecLenf(v2mat, v3mat); | 
					
						
							|  |  |  | 				length[2] = VecLenf(v3mat, v4mat); | 
					
						
							|  |  |  | 				length[3] = VecLenf(v4mat, v1mat); | 
					
						
							|  |  |  | 				sort[0] = ef->e1; | 
					
						
							|  |  |  | 				sort[1] = ef->e2; | 
					
						
							|  |  |  | 				sort[2] = ef->e3; | 
					
						
							|  |  |  | 				sort[3] = ef->e4; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				// Beauty Short Edges
 | 
					
						
							|  |  |  | 				if(beauty & B_BEAUTY_SHORT) { | 
					
						
							|  |  |  | 					for(j=0;j<2;j++) { | 
					
						
							|  |  |  | 						hold = -1; | 
					
						
							|  |  |  | 						for(i=0;i<4;i++) { | 
					
						
							|  |  |  | 							if(length[i] < 0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 								continue; | 
					
						
							|  |  |  | 							} else if(hold == -1) { | 
					
						
							|  |  |  | 								hold = i; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} else { | 
					
						
							|  |  |  | 								if(length[hold] < length[i]) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									hold = i; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2009-04-20 15:06:46 +00:00
										 |  |  | 						if (hold > -1) { | 
					
						
							|  |  |  | 							sort[hold]->f &= ~SELECT; | 
					
						
							|  |  |  | 							sort[hold]->f2 |= EDGENEW; | 
					
						
							|  |  |  | 							length[hold] = -1; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				// Beauty Long Edges
 | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					 for(j=0;j<2;j++) { | 
					
						
							|  |  |  | 						hold = -1; | 
					
						
							|  |  |  | 						for(i=0;i<4;i++) { | 
					
						
							|  |  |  | 							if(length[i] < 0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 								continue; | 
					
						
							|  |  |  | 							} else if(hold == -1) { | 
					
						
							|  |  |  | 								hold = i; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} else { | 
					
						
							|  |  |  | 								if(length[hold] > length[i]) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									hold = i; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2009-04-20 15:06:46 +00:00
										 |  |  | 						if (hold > -1) { | 
					
						
							|  |  |  | 							sort[hold]->f &= ~SELECT; | 
					
						
							|  |  |  | 							sort[hold]->f2 |= EDGENEW; | 
					
						
							|  |  |  | 							length[hold] = -1; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-04-20 15:06:46 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	gh = BLI_ghash_new(BLI_ghashutil_ptrhash, BLI_ghashutil_ptrcmp); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// If we are knifing, We only need the selected edges that were cut, so deselect if it was not cut
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(beauty & B_KNIFE) { | 
					
						
							|  |  |  | 		for(eed= em->edges.first;eed;eed=eed->next) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if( eed->tmp.fp == 0 ) { | 
					
						
							|  |  |  | 				EM_select_edge(eed,0); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// So for each edge, if it is selected, we allocate an array of size cuts+2
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	// so we can have a place for the v1, the new verts and v2
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(eed=em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 		if(eed->f & flag) { | 
					
						
							|  |  |  | 			templist = MEM_mallocN(sizeof(EditVert*)*(numcuts+2),"vertlist"); | 
					
						
							|  |  |  | 			templist[0] = eed->v1; | 
					
						
							|  |  |  | 			for(i=0;i<numcuts;i++) { | 
					
						
							|  |  |  | 				// This function creates the new vert and returns it back
 | 
					
						
							|  |  |  | 				// to the array
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 				templist[i+1] = subdivideedgenum(em, eed, i+1, numcuts, smooth, fractal, beauty); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				//while we are here, we can copy edge info from the original edge
 | 
					
						
							|  |  |  | 				cedge = addedgelist(em, templist[i],templist[i+1],eed); | 
					
						
							|  |  |  | 				// Also set the edge f2 to EDGENEW so that we can use this info later
 | 
					
						
							|  |  |  | 				cedge->f2 = EDGENEW; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			templist[i+1] = eed->v2; | 
					
						
							|  |  |  | 			//Do the last edge too
 | 
					
						
							|  |  |  | 			cedge = addedgelist(em, templist[i],templist[i+1],eed); | 
					
						
							|  |  |  | 			cedge->f2 = EDGENEW; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			// Now that the edge is subdivided, we can put its verts in the ghash
 | 
					
						
							|  |  |  | 			BLI_ghash_insert(gh, eed, templist); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | //	DAG_id_flush_update(obedit->data, OB_RECALC_DATA);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Now for each face in the mesh we need to figure out How many edges were cut
 | 
					
						
							|  |  |  | 	// and which filling method to use for that face
 | 
					
						
							|  |  |  | 	for(ef = em->faces.first;ef;ef = ef->next) { | 
					
						
							|  |  |  | 		edgecount = 0; | 
					
						
							|  |  |  | 		facetype = 3; | 
					
						
							|  |  |  | 		if(ef->e1->f & flag) {edgecount++;} | 
					
						
							|  |  |  | 		if(ef->e2->f & flag) {edgecount++;} | 
					
						
							|  |  |  | 		if(ef->e3->f & flag) {edgecount++;} | 
					
						
							|  |  |  | 		if(ef->v4) { | 
					
						
							|  |  |  | 			facetype = 4; | 
					
						
							|  |  |  | 			if(ef->e4->f & flag) {edgecount++;} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(facetype == 4) { | 
					
						
							|  |  |  | 			switch(edgecount) { | 
					
						
							|  |  |  | 				case 0: | 
					
						
							|  |  |  | 					if(beauty & B_KNIFE && numcuts == 1){ | 
					
						
							|  |  |  | 						/*Test for when knifing through two opposite verts but no edges*/ | 
					
						
							|  |  |  | 						touchcount = 0; | 
					
						
							|  |  |  | 						if(ef->v1->f1) touchcount++; | 
					
						
							|  |  |  | 						if(ef->v2->f1) touchcount++; | 
					
						
							|  |  |  | 						if(ef->v3->f1) touchcount++; | 
					
						
							|  |  |  | 						if(ef->v4->f1) touchcount++; | 
					
						
							|  |  |  | 						if(touchcount == 2){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							if(ef->v1->f1 && ef->v3->f1){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								ef->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 								fill_quad_doublevert(em, ef, 1, 3); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} | 
					
						
							|  |  |  | 							else if(ef->v2->f1 && ef->v4->f1){ | 
					
						
							|  |  |  | 								ef->f1 = SELECT; | 
					
						
							|  |  |  | 								fill_quad_doublevert(em, ef, 2, 4); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				case 1: | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					if(beauty & B_KNIFE && numcuts == 1){ | 
					
						
							|  |  |  | 						/*Test for when knifing through an edge and one vert*/ | 
					
						
							|  |  |  | 						touchcount = 0; | 
					
						
							|  |  |  | 						if(ef->v1->f1) touchcount++; | 
					
						
							|  |  |  | 						if(ef->v2->f1) touchcount++; | 
					
						
							|  |  |  | 						if(ef->v3->f1) touchcount++; | 
					
						
							|  |  |  | 						if(ef->v4->f1) touchcount++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						if(touchcount == 1){ | 
					
						
							|  |  |  | 							if( (ef->e1->f & flag && ( !ef->e1->v1->f1 && !ef->e1->v2->f1 )) || | 
					
						
							|  |  |  | 								(ef->e2->f & flag && ( !ef->e2->v1->f1 && !ef->e2->v2->f1 )) || | 
					
						
							|  |  |  | 								(ef->e3->f & flag && ( !ef->e3->v1->f1 && !ef->e3->v2->f1 )) || | 
					
						
							|  |  |  | 								(ef->e4->f & flag && ( !ef->e4->v1->f1 && !ef->e4->v2->f1 )) ){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 								ef->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								fill_quad_singlevert(em, ef, gh); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else{ | 
					
						
							|  |  |  | 								ef->f1 = SELECT; | 
					
						
							|  |  |  | 								fill_quad_single(em, ef, gh, numcuts, seltype); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						else{ | 
					
						
							|  |  |  | 							ef->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							fill_quad_single(em, ef, gh, numcuts, seltype); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					else{ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						ef->f1 = SELECT; | 
					
						
							|  |  |  | 						fill_quad_single(em, ef, gh, numcuts, seltype); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				case 2: ef->f1 = SELECT; | 
					
						
							|  |  |  | 					// if there are 2, we check if edge 1 and 3 are either both on or off that way
 | 
					
						
							|  |  |  | 					// we can tell if the selected pair is Adjacent or Opposite of each other
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					if((ef->e1->f & flag && ef->e3->f & flag) || | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					   (ef->e2->f & flag && ef->e4->f & flag)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						fill_quad_double_op(em, ef, gh, numcuts); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					}else{ | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 						switch(0) { // XXX scene->toolsettings->cornertype) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							case 0:	fill_quad_double_adj_path(em, ef, gh, numcuts); break; | 
					
						
							|  |  |  | 							case 1:	fill_quad_double_adj_inner(em, ef, gh, numcuts); break; | 
					
						
							|  |  |  | 							case 2:	fill_quad_double_adj_fan(em, ef, gh, numcuts); break; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				case 3: ef->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					fill_quad_triple(em, ef, gh, numcuts); | 
					
						
							|  |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				case 4: ef->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					fill_quad_quadruple(em, ef, gh, numcuts, smooth, fractal, beauty); | 
					
						
							|  |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			switch(edgecount) { | 
					
						
							|  |  |  | 				case 0: break; | 
					
						
							|  |  |  | 				case 1: ef->f1 = SELECT; | 
					
						
							|  |  |  | 					fill_tri_single(em, ef, gh, numcuts, seltype); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				case 2: ef->f1 = SELECT; | 
					
						
							|  |  |  | 					fill_tri_double(em, ef, gh, numcuts); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				case 3: ef->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 					fill_tri_triple(em, ef, gh, numcuts, smooth, fractal, beauty); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Delete Old Edges and Faces
 | 
					
						
							|  |  |  | 	for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 		if(BLI_ghash_haskey(gh,eed)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			eed->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			eed->f1 = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	free_tagged_edges_faces(em, em->edges.first, em->faces.first); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(seltype == SUBDIV_SELECT_ORIG  && !ctrl) { | 
					
						
							|  |  |  | 		/* bugfix: vertex could get flagged as "not-selected"
 | 
					
						
							|  |  |  | 		// solution: clear flags before, not at the same time as setting SELECT flag -dg
 | 
					
						
							|  |  |  | 		*/ | 
					
						
							|  |  |  | 		for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 			if(!(eed->f2 & EDGENEW || eed->f2 & EDGEOLD)) { | 
					
						
							|  |  |  | 				eed->f &= !flag; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				EM_select_edge(eed,0); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 			if(eed->f2 & EDGENEW || eed->f2 & EDGEOLD) { | 
					
						
							|  |  |  | 				eed->f |= flag; | 
					
						
							|  |  |  | 				EM_select_edge(eed,1); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if ((seltype == SUBDIV_SELECT_INNER || seltype == SUBDIV_SELECT_INNER_SEL)|| ctrl) { | 
					
						
							|  |  |  | 		for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 			if(eed->f2 & EDGEINNER) { | 
					
						
							|  |  |  | 				eed->f |= flag; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				EM_select_edge(eed,1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(eed->v1->f & EDGEINNER) eed->v1->f |= SELECT; | 
					
						
							|  |  |  | 				if(eed->v2->f & EDGEINNER) eed->v2->f |= SELECT; | 
					
						
							|  |  |  | 			}else{ | 
					
						
							|  |  |  | 				eed->f &= !flag; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				EM_select_edge(eed,0); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} else if(seltype == SUBDIV_SELECT_LOOPCUT){ | 
					
						
							|  |  |  | 		for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 			if(eed->f2 & DOUBLEOPFILL){ | 
					
						
							|  |  |  | 				eed->f |= flag; | 
					
						
							|  |  |  | 				EM_select_edge(eed,1); | 
					
						
							|  |  |  | 			}else{ | 
					
						
							|  |  |  | 				eed->f &= !flag; | 
					
						
							|  |  |  | 				EM_select_edge(eed,0); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	 if(em->selectmode & SCE_SELECT_VERTEX) { | 
					
						
							|  |  |  | 		 for(eed = em->edges.first;eed;eed = eed->next) { | 
					
						
							|  |  |  | 			if(eed->f & SELECT) { | 
					
						
							|  |  |  | 				eed->v1->f |= SELECT; | 
					
						
							|  |  |  | 				eed->v2->f |= SELECT; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//fix hide flags for edges. First pass, hide edges of hidden faces
 | 
					
						
							|  |  |  | 	for(ef=em->faces.first; ef; ef=ef->next){ | 
					
						
							|  |  |  | 		if(ef->h){ | 
					
						
							|  |  |  | 			ef->e1->h |= 1; | 
					
						
							|  |  |  | 			ef->e2->h |= 1; | 
					
						
							|  |  |  | 			ef->e3->h |= 1; | 
					
						
							|  |  |  | 			if(ef->e4) ef->e4->h |= 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//second pass: unhide edges of visible faces adjacent to hidden faces
 | 
					
						
							|  |  |  | 	for(ef=em->faces.first; ef; ef=ef->next){ | 
					
						
							|  |  |  | 		if(ef->h == 0){ | 
					
						
							|  |  |  | 			ef->e1->h &= ~1; | 
					
						
							|  |  |  | 			ef->e2->h &= ~1; | 
					
						
							|  |  |  | 			ef->e3->h &= ~1; | 
					
						
							|  |  |  | 			if(ef->e4) ef->e4->h &= ~1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Free the ghash and call MEM_freeN on all the value entries to return
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// that memory
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	BLI_ghash_free(gh, NULL, (GHashValFreeFP)MEM_freeN); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EM_selectmode_flush(em); | 
					
						
							|  |  |  | 	for(ef=em->faces.first;ef;ef = ef->next) { | 
					
						
							|  |  |  | 		if(ef->e4) { | 
					
						
							|  |  |  | 			if(  (ef->e1->f & SELECT && ef->e2->f & SELECT) && | 
					
						
							|  |  |  | 			 (ef->e3->f & SELECT && ef->e4->f & SELECT) ) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				ef->f |= SELECT; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			if(  (ef->e1->f & SELECT && ef->e2->f & SELECT) && ef->e3->f & SELECT) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				ef->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	recalc_editnormals(em); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int count_selected_edges(EditEdge *ed) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int totedge = 0; | 
					
						
							|  |  |  | 	while(ed) { | 
					
						
							|  |  |  | 		ed->tmp.p = 0; | 
					
						
							|  |  |  | 		if( ed->f & SELECT ) totedge++; | 
					
						
							|  |  |  | 		ed= ed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return totedge; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* hurms, as if this makes code readable! It's pointerpointer hiding... (ton) */ | 
					
						
							|  |  |  | typedef EditFace *EVPtr; | 
					
						
							|  |  |  | typedef EVPtr EVPTuple[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** builds EVPTuple array efaa of face tuples (in fact pointers to EditFaces)
 | 
					
						
							|  |  |  | 	sharing one edge. | 
					
						
							|  |  |  | 	arguments: selected edge list, face list. | 
					
						
							|  |  |  | 	Edges will also be tagged accordingly (see eed->f2)		  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int collect_quadedges(EVPTuple *efaa, EditEdge *eed, EditFace *efa) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *e1, *e2, *e3; | 
					
						
							|  |  |  | 	EVPtr *evp; | 
					
						
							|  |  |  | 	int i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* run through edges, if selected, set pointer edge-> facearray */ | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		eed->f2= 0; | 
					
						
							|  |  |  | 		eed->f1= 0; | 
					
						
							|  |  |  | 		if( eed->f & SELECT ) { | 
					
						
							|  |  |  | 			eed->tmp.p = (EditVert *) (&efaa[i]); | 
					
						
							|  |  |  | 			i++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else eed->tmp.p = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* find edges pointing to 2 faces by procedure:
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	- run through faces and their edges, increase | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	  face counter e->f1 for each face | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		efa->f1= 0; | 
					
						
							|  |  |  | 		if(efa->v4==0 && (efa->f & SELECT)) {  /* if selected triangle */ | 
					
						
							|  |  |  | 			e1= efa->e1; | 
					
						
							|  |  |  | 			e2= efa->e2; | 
					
						
							|  |  |  | 			e3= efa->e3; | 
					
						
							|  |  |  | 			if(e1->f2<3 && e1->tmp.p) { | 
					
						
							|  |  |  | 				if(e1->f2<2) { | 
					
						
							|  |  |  | 					evp= (EVPtr *) e1->tmp.p; | 
					
						
							|  |  |  | 					evp[(int)e1->f2] = efa; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				e1->f2+= 1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(e2->f2<3 && e2->tmp.p) { | 
					
						
							|  |  |  | 				if(e2->f2<2) { | 
					
						
							|  |  |  | 					evp= (EVPtr *) e2->tmp.p; | 
					
						
							|  |  |  | 					evp[(int)e2->f2]= efa; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				e2->f2+= 1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(e3->f2<3 && e3->tmp.p) { | 
					
						
							|  |  |  | 				if(e3->f2<2) { | 
					
						
							|  |  |  | 					evp= (EVPtr *) e3->tmp.p; | 
					
						
							|  |  |  | 					evp[(int)e3->f2]= efa; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				e3->f2+= 1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			/* set to 3 to make sure these are not flipped or joined */ | 
					
						
							|  |  |  | 			efa->e1->f2= 3; | 
					
						
							|  |  |  | 			efa->e2->f2= 3; | 
					
						
							|  |  |  | 			efa->e3->f2= 3; | 
					
						
							|  |  |  | 			if (efa->e4) efa->e4->f2= 3; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		efa= efa->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return i; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | /* returns vertices of two adjacent triangles forming a quad
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  |    - can be righthand or lefthand | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			4-----3 | 
					
						
							|  |  |  | 			|\	| | 
					
						
							|  |  |  | 			| \ 2 | <- efa1 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			|  \  | | 
					
						
							|  |  |  | 	  efa-> | 1 \ | | 
					
						
							|  |  |  | 			|	\| | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			1-----2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | #define VTEST(face, num, other) \
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	(face->v##num != other->v1 && face->v##num != other->v2 && face->v##num != other->v3) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void givequadverts(EditFace *efa, EditFace *efa1, EditVert **v1, EditVert **v2, EditVert **v3, EditVert **v4, int *vindex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if VTEST(efa, 1, efa1) { | 
					
						
							|  |  |  | 		*v1= efa->v1; | 
					
						
							|  |  |  | 		*v2= efa->v2; | 
					
						
							|  |  |  | 		vindex[0]= 0; | 
					
						
							|  |  |  | 		vindex[1]= 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if VTEST(efa, 2, efa1) { | 
					
						
							|  |  |  | 		*v1= efa->v2; | 
					
						
							|  |  |  | 		*v2= efa->v3; | 
					
						
							|  |  |  | 		vindex[0]= 1; | 
					
						
							|  |  |  | 		vindex[1]= 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if VTEST(efa, 3, efa1) { | 
					
						
							|  |  |  | 		*v1= efa->v3; | 
					
						
							|  |  |  | 		*v2= efa->v1; | 
					
						
							|  |  |  | 		vindex[0]= 2; | 
					
						
							|  |  |  | 		vindex[1]= 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if VTEST(efa1, 1, efa) { | 
					
						
							|  |  |  | 		*v3= efa1->v1; | 
					
						
							|  |  |  | 		*v4= efa1->v2; | 
					
						
							|  |  |  | 		vindex[2]= 0; | 
					
						
							|  |  |  | 		vindex[3]= 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if VTEST(efa1, 2, efa) { | 
					
						
							|  |  |  | 		*v3= efa1->v2; | 
					
						
							|  |  |  | 		*v4= efa1->v3; | 
					
						
							|  |  |  | 		vindex[2]= 1; | 
					
						
							|  |  |  | 		vindex[3]= 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if VTEST(efa1, 3, efa) { | 
					
						
							|  |  |  | 		*v3= efa1->v3; | 
					
						
							|  |  |  | 		*v4= efa1->v1; | 
					
						
							|  |  |  | 		vindex[2]= 2; | 
					
						
							|  |  |  | 		vindex[3]= 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		*v3= *v4= NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Helper functions for edge/quad edit features*/ | 
					
						
							|  |  |  | static void untag_edges(EditFace *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	f->e1->f1 = 0; | 
					
						
							|  |  |  | 	f->e2->f1 = 0; | 
					
						
							|  |  |  | 	f->e3->f1 = 0; | 
					
						
							|  |  |  | 	if (f->e4) f->e4->f1 = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** remove and free list of tagged edges and faces */ | 
					
						
							|  |  |  | static void free_tagged_edges_faces(EditMesh *em, EditEdge *eed, EditFace *efa) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *nexted; | 
					
						
							|  |  |  | 	EditFace *nextvl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		nextvl= efa->next; | 
					
						
							|  |  |  | 		if(efa->f1) { | 
					
						
							|  |  |  | 			BLI_remlink(&em->faces, efa); | 
					
						
							|  |  |  | 			free_editface(em, efa); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			/* avoid deleting edges that are still in use */ | 
					
						
							|  |  |  | 			untag_edges(efa); | 
					
						
							|  |  |  | 		efa= nextvl; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		nexted= eed->next; | 
					
						
							|  |  |  | 		if(eed->f1) { | 
					
						
							|  |  |  | 			remedge(em, eed); | 
					
						
							|  |  |  | 			free_editedge(em, eed); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= nexted; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ******************** BEGIN TRIANGLE TO QUAD ************************************* */ | 
					
						
							|  |  |  | static float measure_facepair(EditVert *v1, EditVert *v2, EditVert *v3, EditVert *v4, float limit) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*gives a 'weight' to a pair of triangles that join an edge to decide how good a join they would make*/ | 
					
						
							|  |  |  | 	/*Note: this is more complicated than it needs to be and should be cleaned up...*/ | 
					
						
							|  |  |  | 	float	measure = 0.0, noA1[3], noA2[3], noB1[3], noB2[3], normalADiff, normalBDiff, | 
					
						
							|  |  |  | 			edgeVec1[3], edgeVec2[3], edgeVec3[3], edgeVec4[3], diff, | 
					
						
							|  |  |  | 			minarea, maxarea, areaA, areaB; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*First Test: Normal difference*/ | 
					
						
							|  |  |  | 	CalcNormFloat(v1->co, v2->co, v3->co, noA1); | 
					
						
							|  |  |  | 	CalcNormFloat(v1->co, v3->co, v4->co, noA2); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(noA1[0] == noA2[0] && noA1[1] == noA2[1] && noA1[2] == noA2[2]) normalADiff = 0.0; | 
					
						
							| 
									
										
										
										
											2009-09-18 15:47:00 +00:00
										 |  |  | 	else normalADiff = RAD2DEG(VecAngle2(noA1, noA2)); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		//if(!normalADiff) normalADiff = 179;
 | 
					
						
							|  |  |  | 	CalcNormFloat(v2->co, v3->co, v4->co, noB1); | 
					
						
							|  |  |  | 	CalcNormFloat(v4->co, v1->co, v2->co, noB2); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(noB1[0] == noB2[0] && noB1[1] == noB2[1] && noB1[2] == noB2[2]) normalBDiff = 0.0; | 
					
						
							| 
									
										
										
										
											2009-09-18 15:47:00 +00:00
										 |  |  | 	else normalBDiff = RAD2DEG(VecAngle2(noB1, noB2)); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		//if(!normalBDiff) normalBDiff = 179;
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	measure += (normalADiff/360) + (normalBDiff/360); | 
					
						
							|  |  |  | 	if(measure > limit) return measure; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*Second test: Colinearity*/ | 
					
						
							|  |  |  | 	VecSubf(edgeVec1, v1->co, v2->co); | 
					
						
							|  |  |  | 	VecSubf(edgeVec2, v2->co, v3->co); | 
					
						
							|  |  |  | 	VecSubf(edgeVec3, v3->co, v4->co); | 
					
						
							|  |  |  | 	VecSubf(edgeVec4, v4->co, v1->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	diff = 0.0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	diff = ( | 
					
						
							| 
									
										
										
										
											2009-09-18 15:47:00 +00:00
										 |  |  | 		fabs(RAD2DEG(VecAngle2(edgeVec1, edgeVec2)) - 90) + | 
					
						
							|  |  |  | 		fabs(RAD2DEG(VecAngle2(edgeVec2, edgeVec3)) - 90) + | 
					
						
							|  |  |  | 		fabs(RAD2DEG(VecAngle2(edgeVec3, edgeVec4)) - 90) + | 
					
						
							|  |  |  | 		fabs(RAD2DEG(VecAngle2(edgeVec4, edgeVec1)) - 90)) / 360; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!diff) return 0.0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	measure +=  diff; | 
					
						
							|  |  |  | 	if(measure > limit) return measure; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*Third test: Concavity*/ | 
					
						
							|  |  |  | 	areaA = AreaT3Dfl(v1->co, v2->co, v3->co) + AreaT3Dfl(v1->co, v3->co, v4->co); | 
					
						
							|  |  |  | 	areaB = AreaT3Dfl(v2->co, v3->co, v4->co) + AreaT3Dfl(v4->co, v1->co, v2->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(areaA <= areaB) minarea = areaA; | 
					
						
							|  |  |  | 	else minarea = areaB; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(areaA >= areaB) maxarea = areaA; | 
					
						
							|  |  |  | 	else maxarea = areaB; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!maxarea) measure += 1; | 
					
						
							|  |  |  | 	else measure += (1 - (minarea / maxarea)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return measure; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define T2QUV_LIMIT 0.005
 | 
					
						
							|  |  |  | #define T2QCOL_LIMIT 3
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | static int compareFaceAttribs(EditMesh *em, EditFace *f1, EditFace *f2, EditEdge *eed) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	/*Test to see if the per-face attributes for the joining edge match within limit*/ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MTFace *tf1, *tf2; | 
					
						
							|  |  |  | 	unsigned int *col1, *col2; | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	short i,attrok=0, flag = 0, /* XXX scene->toolsettings->editbutflag,*/ fe1[2], fe2[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	tf1 = CustomData_em_get(&em->fdata, f1->data, CD_MTFACE); | 
					
						
							|  |  |  | 	tf2 = CustomData_em_get(&em->fdata, f2->data, CD_MTFACE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	col1 = CustomData_em_get(&em->fdata, f1->data, CD_MCOL); | 
					
						
							|  |  |  | 	col2 = CustomData_em_get(&em->fdata, f2->data, CD_MCOL); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*store indices for faceedges*/ | 
					
						
							|  |  |  | 	f1->v1->f1 = 0; | 
					
						
							|  |  |  | 	f1->v2->f1 = 1; | 
					
						
							|  |  |  | 	f1->v3->f1 = 2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	fe1[0] = eed->v1->f1; | 
					
						
							|  |  |  | 	fe1[1] = eed->v2->f1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	f2->v1->f1 = 0; | 
					
						
							|  |  |  | 	f2->v2->f1 = 1; | 
					
						
							|  |  |  | 	f2->v3->f1 = 2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	fe2[0] = eed->v1->f1; | 
					
						
							|  |  |  | 	fe2[1] = eed->v2->f1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*compare faceedges for each face attribute. Additional per face attributes can be added later*/ | 
					
						
							|  |  |  | 	/*do UVs*/ | 
					
						
							|  |  |  | 	if(flag & B_JOINTRIA_UV){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(tf1 == NULL || tf2 == NULL) attrok |= B_JOINTRIA_UV; | 
					
						
							|  |  |  | 		else if(tf1->tpage != tf2->tpage); /*do nothing*/ | 
					
						
							|  |  |  | 		else{ | 
					
						
							|  |  |  | 			for(i = 0; i < 2; i++){ | 
					
						
							|  |  |  | 				if(tf1->uv[fe1[i]][0] + T2QUV_LIMIT > tf2->uv[fe2[i]][0] && tf1->uv[fe1[i]][0] - T2QUV_LIMIT < tf2->uv[fe2[i]][0] && | 
					
						
							|  |  |  | 					tf1->uv[fe1[i]][1] + T2QUV_LIMIT > tf2->uv[fe2[i]][1] && tf1->uv[fe1[i]][1] - T2QUV_LIMIT < tf2->uv[fe2[i]][1]) attrok |= B_JOINTRIA_UV; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*do VCOLs*/ | 
					
						
							|  |  |  | 	if(flag & B_JOINTRIA_VCOL){ | 
					
						
							|  |  |  | 		if(!col1 || !col2) attrok |= B_JOINTRIA_VCOL; | 
					
						
							|  |  |  | 		else{ | 
					
						
							|  |  |  | 			char *f1vcol, *f2vcol; | 
					
						
							|  |  |  | 			for(i = 0; i < 2; i++){ | 
					
						
							|  |  |  | 				f1vcol = (char *)&(col1[fe1[i]]); | 
					
						
							|  |  |  | 				f2vcol = (char *)&(col2[fe2[i]]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				/*compare f1vcol with f2vcol*/ | 
					
						
							|  |  |  | 				if(	f1vcol[1] + T2QCOL_LIMIT > f2vcol[1] && f1vcol[1] - T2QCOL_LIMIT < f2vcol[1] && | 
					
						
							|  |  |  | 					f1vcol[2] + T2QCOL_LIMIT > f2vcol[2] && f1vcol[2] - T2QCOL_LIMIT < f2vcol[2] && | 
					
						
							|  |  |  | 					f1vcol[3] + T2QCOL_LIMIT > f2vcol[3] && f1vcol[3] - T2QCOL_LIMIT < f2vcol[3]) attrok |= B_JOINTRIA_VCOL; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if( ((attrok & B_JOINTRIA_UV) == (flag & B_JOINTRIA_UV)) && ((attrok & B_JOINTRIA_VCOL) == (flag & B_JOINTRIA_VCOL)) ) return 1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | static int fplcmp(const void *v1, const void *v2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const EditEdge *e1= *((EditEdge**)v1), *e2=*((EditEdge**)v2); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if( e1->crease > e2->crease) return 1; | 
					
						
							|  |  |  | 	else if( e1->crease < e2->crease) return -1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*Bitflags for edges.*/ | 
					
						
							|  |  |  | #define T2QDELETE	1
 | 
					
						
							|  |  |  | #define T2QCOMPLEX	2
 | 
					
						
							|  |  |  | #define T2QJOIN		4
 | 
					
						
							|  |  |  | void join_triangles(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *v1, *v2, *v3, *v4, *eve; | 
					
						
							|  |  |  | 	EditEdge *eed, **edsortblock = NULL, **edb = NULL; | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	EVPTuple *efaar = NULL; | 
					
						
							|  |  |  | 	EVPtr *efaa = NULL; | 
					
						
							|  |  |  | 	float *creases = NULL; | 
					
						
							|  |  |  | 	float measure; /*Used to set tolerance*/ | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	float limit = 0.0f; // XXX scene->toolsettings->jointrilimit;
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	int i, ok, totedge=0, totseledge=0, complexedges, vindex[4]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*if we take a long time on very dense meshes we want waitcursor to display*/ | 
					
						
							|  |  |  | 	waitcursor(1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	totseledge = count_selected_edges(em->edges.first); | 
					
						
							|  |  |  | 	if(totseledge==0) return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*abusing crease value to store weights for edge pairs. Nasty*/ | 
					
						
							|  |  |  | 	for(eed=em->edges.first; eed; eed=eed->next) totedge++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(totedge) creases = MEM_callocN(sizeof(float) * totedge, "Join Triangles Crease Array"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(eed=em->edges.first, i = 0; eed; eed=eed->next, i++){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		creases[i] = eed->crease; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		eed->crease = 0.0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*clear temp flags*/ | 
					
						
							|  |  |  | 	for(eve=em->verts.first; eve; eve=eve->next) eve->f1 = eve->f2 = 0; | 
					
						
							|  |  |  | 	for(eed=em->edges.first; eed; eed=eed->next) eed->f2 = eed->f1 = 0; | 
					
						
							|  |  |  | 	for(efa=em->faces.first; efa; efa=efa->next) efa->f1 = efa->tmp.l = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*For every selected 2 manifold edge, create pointers to its two faces.*/ | 
					
						
							|  |  |  | 	efaar= (EVPTuple *) MEM_callocN(totseledge * sizeof(EVPTuple), "Tri2Quad"); | 
					
						
							|  |  |  | 	ok = collect_quadedges(efaar, em->edges.first, em->faces.first); | 
					
						
							|  |  |  | 	complexedges = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(ok){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/*clear tmp.l flag and store number of faces that are selected and coincident to current face here.*/ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 			/* eed->f2 is 2 only if this edge is part of exactly two
 | 
					
						
							|  |  |  | 			   triangles, and both are selected, and it has EVPTuple assigned */ | 
					
						
							|  |  |  | 			if(eed->f2 == 2){ | 
					
						
							|  |  |  | 				efaa= (EVPtr *) eed->tmp.p; | 
					
						
							|  |  |  | 				efaa[0]->tmp.l++; | 
					
						
							|  |  |  | 				efaa[1]->tmp.l++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 			if(eed->f2 == 2){ | 
					
						
							|  |  |  | 				efaa= (EVPtr *) eed->tmp.p; | 
					
						
							|  |  |  | 				v1 = v2 = v3 = v4 = NULL; | 
					
						
							|  |  |  | 				givequadverts(efaa[0], efaa[1], &v1, &v2, &v3, &v4, vindex); | 
					
						
							|  |  |  | 				if(v1 && v2 && v3 && v4){ | 
					
						
							|  |  |  | 					/*test if simple island first. This mimics 2.42 behaviour and the tests are less restrictive.*/ | 
					
						
							|  |  |  | 					if(efaa[0]->tmp.l == 1 && efaa[1]->tmp.l == 1){ | 
					
						
							| 
									
										
										
										
											2009-04-20 15:06:46 +00:00
										 |  |  | 						eed->f1 |= T2QJOIN; | 
					
						
							|  |  |  | 						efaa[0]->f1 = 1; //mark for join
 | 
					
						
							|  |  |  | 						efaa[1]->f1 = 1; //mark for join
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					else{ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						/*	The face pair is part of a 'complex' island, so the rules for dealing with it are more involved.
 | 
					
						
							|  |  |  | 							Depending on what options the user has chosen, this face pair can be 'thrown out' based upon the following criteria: | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							1: the two faces do not share the same material | 
					
						
							|  |  |  | 							2: the edge joining the two faces is marked as sharp. | 
					
						
							|  |  |  | 							3: the two faces UV's do not make a good match | 
					
						
							|  |  |  | 							4: the two faces Vertex colors do not make a good match | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							If the face pair passes all the applicable tests, it is then given a 'weight' with the measure_facepair() function. | 
					
						
							|  |  |  | 							This measures things like concavity, colinearity ect. If this weight is below the threshold set by the user | 
					
						
							|  |  |  | 							the edge joining them is marked as being 'complex' and will be compared against other possible pairs which contain one of the | 
					
						
							|  |  |  | 							same faces in the current pair later. | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							This technique is based upon an algorithm that Campbell Barton developed for his Tri2Quad script that was previously part of | 
					
						
							|  |  |  | 							the python scripts bundled with Blender releases. | 
					
						
							|  |  |  | 						*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | // XXX						if(scene->toolsettings->editbutflag & B_JOINTRIA_SHARP && eed->sharp); /*do nothing*/
 | 
					
						
							|  |  |  | //						else if(scene->toolsettings->editbutflag & B_JOINTRIA_MAT && efaa[0]->mat_nr != efaa[1]->mat_nr); /*do nothing*/
 | 
					
						
							|  |  |  | //						else if(((scene->toolsettings->editbutflag & B_JOINTRIA_UV) || (scene->toolsettings->editbutflag & B_JOINTRIA_VCOL)) &&
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						compareFaceAttribs(em, efaa[0], efaa[1], eed); // XXX == 0); /*do nothing*/
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | //						else{
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							measure = measure_facepair(v1, v2, v3, v4, limit); | 
					
						
							|  |  |  | 							if(measure < limit){ | 
					
						
							|  |  |  | 								complexedges++; | 
					
						
							|  |  |  | 								eed->f1 |= T2QCOMPLEX; | 
					
						
							|  |  |  | 								eed->crease = measure; /*we dont mark edges for join yet*/ | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | //						}
 | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/*Quicksort the complex edges according to their weighting*/ | 
					
						
							|  |  |  | 		if(complexedges){ | 
					
						
							|  |  |  | 			edsortblock = edb = MEM_callocN(sizeof(EditEdge*) * complexedges, "Face Pairs quicksort Array"); | 
					
						
							|  |  |  | 			for(eed = em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 				if(eed->f1 & T2QCOMPLEX){ | 
					
						
							|  |  |  | 					*edb = eed; | 
					
						
							|  |  |  | 					edb++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			qsort(edsortblock, complexedges, sizeof(EditEdge*), fplcmp); | 
					
						
							|  |  |  | 			/*now go through and mark the edges who get the highest weighting*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			for(edb=edsortblock, i=0; i < complexedges; edb++, i++){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				efaa = (EVPtr *)((*edb)->tmp.p); /*suspect!*/ | 
					
						
							|  |  |  | 				if( !efaa[0]->f1 && !efaa[1]->f1){ | 
					
						
							|  |  |  | 					efaa[0]->f1 = 1; //mark for join
 | 
					
						
							|  |  |  | 					efaa[1]->f1 = 1; //mark for join
 | 
					
						
							|  |  |  | 					(*edb)->f1 |= T2QJOIN; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/*finally go through all edges marked for join (simple and complex) and create new faces*/ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 			if(eed->f1 & T2QJOIN){ | 
					
						
							|  |  |  | 				efaa= (EVPtr *)eed->tmp.p; | 
					
						
							|  |  |  | 				v1 = v2 = v3 = v4 = NULL; | 
					
						
							|  |  |  | 				givequadverts(efaa[0], efaa[1], &v1, &v2, &v3, &v4, vindex); | 
					
						
							|  |  |  | 				if((v1 && v2 && v3 && v4) && (exist_face(em, v1, v2, v3, v4)==0)){ /*exist_face is very slow! Needs to be adressed.*/ | 
					
						
							|  |  |  | 					/*flag for delete*/ | 
					
						
							|  |  |  | 					eed->f1 |= T2QDELETE; | 
					
						
							|  |  |  | 					/*create new quad and select*/ | 
					
						
							|  |  |  | 					efa = EM_face_from_faces(em, efaa[0], efaa[1], vindex[0], vindex[1], 4+vindex[2], 4+vindex[3]); | 
					
						
							|  |  |  | 					EM_select_face(efa,1); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else{ | 
					
						
							|  |  |  | 						efaa[0]->f1 = 0; | 
					
						
							|  |  |  | 						efaa[1]->f1 = 0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*free data and cleanup*/ | 
					
						
							|  |  |  | 	if(creases){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		for(eed=em->edges.first, i = 0; eed; eed=eed->next, i++) eed->crease = creases[i]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		MEM_freeN(creases); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 		if(eed->f1 & T2QDELETE) eed->f1 = 1; | 
					
						
							|  |  |  | 		else eed->f1 = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	free_tagged_edges_faces(em, em->edges.first, em->faces.first); | 
					
						
							|  |  |  | 	if(efaar) MEM_freeN(efaar); | 
					
						
							|  |  |  | 	if(edsortblock) MEM_freeN(edsortblock); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EM_selectmode_flush(em); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | /* ******************** END TRIANGLE TO QUAD ************************************* */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define FACE_MARKCLEAR(f) (f->f1 = 1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* quick hack, basically a copy of beauty_fill */ | 
					
						
							|  |  |  | void edge_flip(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *v1, *v2, *v3, *v4; | 
					
						
							|  |  |  | 	EditEdge *eed, *nexted; | 
					
						
							|  |  |  | 	EditFace *efa, *w; | 
					
						
							|  |  |  | 	//void **efaar, **efaa;
 | 
					
						
							|  |  |  | 	EVPTuple *efaar; | 
					
						
							|  |  |  | 	EVPtr *efaa; | 
					
						
							|  |  |  | 	int totedge, ok, vindex[4]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* - all selected edges with two faces
 | 
					
						
							|  |  |  | 	 * - find the faces: store them in edges (using datablock) | 
					
						
							|  |  |  | 	 * - per edge: - test convex | 
					
						
							|  |  |  | 	 *			   - test edge: flip? | 
					
						
							|  |  |  | 						- if true: remedge,  addedge, all edges at the edge get new face pointers | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	EM_selectmode_flush(em);	// makes sure in selectmode 'face' the edges of selected faces are selected too
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	totedge = count_selected_edges(em->edges.first); | 
					
						
							|  |  |  | 	if(totedge==0) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* temporary array for : edge -> face[1], face[2] */ | 
					
						
							|  |  |  | 	efaar= (EVPTuple *) MEM_callocN(totedge * sizeof(EVPTuple), "edgeflip"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ok = collect_quadedges(efaar, em->edges.first, em->faces.first); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	eed= em->edges.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		nexted= eed->next; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eed->f2==2) {  /* points to 2 faces */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			efaa= (EVPtr *) eed->tmp.p; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			/* don't do it if flagged */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ok= 1; | 
					
						
							|  |  |  | 			efa= efaa[0]; | 
					
						
							|  |  |  | 			if(efa->e1->f1 || efa->e2->f1 || efa->e3->f1) ok= 0; | 
					
						
							|  |  |  | 			efa= efaa[1]; | 
					
						
							|  |  |  | 			if(efa->e1->f1 || efa->e2->f1 || efa->e3->f1) ok= 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(ok) { | 
					
						
							|  |  |  | 				/* test convex */ | 
					
						
							|  |  |  | 				givequadverts(efaa[0], efaa[1], &v1, &v2, &v3, &v4, vindex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | 		4-----3		4-----3 | 
					
						
							|  |  |  | 		|\	|		|	/| | 
					
						
							|  |  |  | 		| \ 1 |		| 1 / | | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		|  \  |  ->	|  /  | | 
					
						
							|  |  |  | 		| 0 \ |		| / 0 | | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		|	\|		|/	| | 
					
						
							|  |  |  | 		1-----2		1-----2 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 				/* make new faces */ | 
					
						
							|  |  |  | 				if (v1 && v2 && v3) { | 
					
						
							|  |  |  | 					if( convex(v1->co, v2->co, v3->co, v4->co) ) { | 
					
						
							|  |  |  | 						if(exist_face(em, v1, v2, v3, v4)==0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							/* outch this may break seams */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							w= EM_face_from_faces(em, efaa[0], efaa[1], vindex[0], | 
					
						
							|  |  |  | 								vindex[1], 4+vindex[2], -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							EM_select_face(w, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							/* outch this may break seams */ | 
					
						
							|  |  |  | 							w= EM_face_from_faces(em, efaa[0], efaa[1], vindex[0], | 
					
						
							|  |  |  | 								4+vindex[2], 4+vindex[3], -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							EM_select_face(w, 1); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						/* tag as to-be-removed */ | 
					
						
							|  |  |  | 						FACE_MARKCLEAR(efaa[1]); | 
					
						
							|  |  |  | 						FACE_MARKCLEAR(efaa[0]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						eed->f1 = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} /* endif test convex */ | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= nexted; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* clear tagged edges and faces: */ | 
					
						
							|  |  |  | 	free_tagged_edges_faces(em, em->edges.first, em->faces.first); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MEM_freeN(efaar); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | #define DIRECTION_CW	1
 | 
					
						
							|  |  |  | #define DIRECTION_CCW	2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const EnumPropertyItem direction_items[]= { | 
					
						
							|  |  |  | 	{DIRECTION_CW, "CW", 0, "Clockwise", ""}, | 
					
						
							|  |  |  | 	{DIRECTION_CCW, "CCW", 0, "Counter Clockwise", ""}, | 
					
						
							|  |  |  | 	{0, NULL, 0, NULL, NULL}}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define AXIS_X		1
 | 
					
						
							|  |  |  | #define AXIS_Y		2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const EnumPropertyItem axis_items[]= { | 
					
						
							|  |  |  | 	{AXIS_X, "X", 0, "X", ""}, | 
					
						
							|  |  |  | 	{AXIS_Y, "Y", 0, "Y", ""}, | 
					
						
							|  |  |  | 	{0, NULL, 0, NULL, NULL}}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void edge_rotate(EditMesh *em, wmOperator *op, EditEdge *eed, int dir) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert **verts[2]; | 
					
						
							|  |  |  | 	EditFace *face[2], *efa, *newFace[2]; | 
					
						
							|  |  |  | 	EditEdge **edges[2], **hiddenedges, *srchedge; | 
					
						
							|  |  |  | 	int facecount, p1, p2, p3, p4, fac1, fac2, i, j; | 
					
						
							|  |  |  | 	int numhidden, numshared, p[2][4]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* check to make sure that the edge is only part of 2 faces */ | 
					
						
							|  |  |  | 	facecount = 0; | 
					
						
							|  |  |  | 	for(efa = em->faces.first;efa;efa = efa->next) { | 
					
						
							|  |  |  | 		if((efa->e1 == eed || efa->e2 == eed) || (efa->e3 == eed || efa->e4 == eed)) { | 
					
						
							|  |  |  | 			if(facecount >= 2) { | 
					
						
							|  |  |  | 				/* more than two faces with this edge */ | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				face[facecount] = efa; | 
					
						
							|  |  |  | 				facecount++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(facecount < 2) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* how many edges does each face have */ | 
					
						
							|  |  |  |  	if(face[0]->e4) fac1= 4; | 
					
						
							|  |  |  | 	else fac1= 3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(face[1]->e4) fac2= 4; | 
					
						
							|  |  |  | 	else fac2= 3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* make a handy array for verts and edges */ | 
					
						
							|  |  |  | 	verts[0]= &face[0]->v1; | 
					
						
							|  |  |  | 	edges[0]= &face[0]->e1; | 
					
						
							|  |  |  | 	verts[1]= &face[1]->v1; | 
					
						
							|  |  |  | 	edges[1]= &face[1]->e1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* we don't want to rotate edges between faces that share more than one edge */ | 
					
						
							|  |  |  | 	numshared= 0; | 
					
						
							|  |  |  | 	for(i=0; i<fac1; i++) | 
					
						
							|  |  |  | 		for(j=0; j<fac2; j++) | 
					
						
							|  |  |  | 			if (edges[0][i] == edges[1][j]) | 
					
						
							|  |  |  | 				numshared++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(numshared > 1) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* we want to construct an array of vertex indicis in both faces, starting at
 | 
					
						
							|  |  |  | 	   the last vertex of the edge being rotated. | 
					
						
							|  |  |  | 	   - first we find the two vertices that lie on the rotating edge | 
					
						
							|  |  |  | 	   - then we make sure they are ordered according to the face vertex order | 
					
						
							|  |  |  | 	   - and then we construct the array */ | 
					
						
							|  |  |  | 	p1= p2= p3= p4= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(i=0; i<4; i++) { | 
					
						
							|  |  |  | 		if(eed->v1 == verts[0][i]) p1 = i; | 
					
						
							|  |  |  | 		if(eed->v2 == verts[0][i]) p2 = i; | 
					
						
							|  |  |  | 		if(eed->v1 == verts[1][i]) p3 = i; | 
					
						
							|  |  |  | 		if(eed->v2 == verts[1][i]) p4 = i; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if((p1+1)%fac1 == p2) | 
					
						
							|  |  |  | 		SWAP(int, p1, p2); | 
					
						
							|  |  |  | 	if((p3+1)%fac2 == p4) | 
					
						
							|  |  |  | 		SWAP(int, p3, p4); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for (i = 0; i < 4; i++) { | 
					
						
							|  |  |  | 		p[0][i]= (p1 + i)%fac1; | 
					
						
							|  |  |  | 		p[1][i]= (p3 + i)%fac2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create an Array of the Edges who have h set prior to rotate */ | 
					
						
							|  |  |  | 	numhidden = 0; | 
					
						
							|  |  |  | 	for(srchedge = em->edges.first;srchedge;srchedge = srchedge->next) | 
					
						
							|  |  |  | 		if(srchedge->h && ((srchedge->v1->f & SELECT) || (srchedge->v2->f & SELECT))) | 
					
						
							|  |  |  | 			numhidden++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	hiddenedges = MEM_mallocN(sizeof(EditVert*)*numhidden+1, "RotateEdgeHiddenVerts"); | 
					
						
							|  |  |  | 	if(!hiddenedges) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  |         BKE_report(op->reports, RPT_ERROR, "Memory allocation failed"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     numhidden = 0; | 
					
						
							|  |  |  | 	for(srchedge=em->edges.first; srchedge; srchedge=srchedge->next) | 
					
						
							|  |  |  | 		if(srchedge->h && (srchedge->v1->f & SELECT || srchedge->v2->f & SELECT)) | 
					
						
							|  |  |  | 			hiddenedges[numhidden++] = srchedge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create the 2 new faces */ | 
					
						
							|  |  |  | 	if(fac1 == 3 && fac2 == 3) { | 
					
						
							|  |  |  | 		/* no need of reverse setup */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][1], p[0][2], 4+p[1][1], -1); | 
					
						
							|  |  |  | 		newFace[1]= EM_face_from_faces(em, face[1], face[0], p[1][1], p[1][2], 4+p[0][1], -1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(fac1 == 4 && fac2 == 3) { | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		if(dir == DIRECTION_CW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][1], p[0][2], p[0][3], 4+p[1][1]); | 
					
						
							|  |  |  | 			newFace[1]= EM_face_from_faces(em, face[1], face[0], p[1][1], p[1][2], 4+p[0][1], -1); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		} else if (dir == DIRECTION_CCW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][2], 4+p[1][1], p[0][0], p[0][1]); | 
					
						
							|  |  |  | 			newFace[1]= EM_face_from_faces(em, face[1], face[0], 4+p[0][2], p[1][0], p[1][1], -1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			verts[0][p[0][2]]->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			verts[1][p[1][1]]->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(fac1 == 3 && fac2 == 4) { | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		if(dir == DIRECTION_CW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][1], p[0][2], 4+p[1][1], -1); | 
					
						
							|  |  |  | 			newFace[1]= EM_face_from_faces(em, face[1], face[0], p[1][1], p[1][2], p[1][3], 4+p[0][1]); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		} else if (dir == DIRECTION_CCW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][0], p[0][1], 4+p[1][2], -1); | 
					
						
							|  |  |  | 			newFace[1]= EM_face_from_faces(em, face[1], face[0], p[1][1], p[1][2], 4+p[0][1], 4+p[0][2]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			verts[0][p[0][1]]->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			verts[1][p[1][2]]->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else if(fac1 == 4 && fac2 == 4) { | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		if(dir == DIRECTION_CW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][1], p[0][2], p[0][3], 4+p[1][1]); | 
					
						
							|  |  |  | 			newFace[1]= EM_face_from_faces(em, face[1], face[0], p[1][1], p[1][2], p[1][3], 4+p[0][1]); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		} else if (dir == DIRECTION_CCW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newFace[0]= EM_face_from_faces(em, face[0], face[1], p[0][2], p[0][3], 4+p[1][1], 4+p[1][2]); | 
					
						
							|  |  |  | 			newFace[1]= EM_face_from_faces(em, face[1], face[0], p[1][2], p[1][3], 4+p[0][1], 4+p[0][2]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			verts[0][p[0][2]]->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			verts[1][p[1][2]]->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		return; /* This should never happen */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	if(dir == DIRECTION_CW || (fac1 == 3 && fac2 == 3)) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		verts[0][p[0][1]]->f |= SELECT; | 
					
						
							|  |  |  | 		verts[1][p[1][1]]->f |= SELECT; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* copy old edge's flags to new center edge*/ | 
					
						
							|  |  |  | 	for(srchedge=em->edges.first;srchedge;srchedge=srchedge->next) { | 
					
						
							|  |  |  | 		if((srchedge->v1->f & SELECT) && (srchedge->v2->f & SELECT)) { | 
					
						
							|  |  |  | 			srchedge->f = eed->f; | 
					
						
							|  |  |  | 			srchedge->h = eed->h; | 
					
						
							|  |  |  | 			srchedge->dir = eed->dir; | 
					
						
							|  |  |  | 			srchedge->seam = eed->seam; | 
					
						
							|  |  |  | 			srchedge->crease = eed->crease; | 
					
						
							|  |  |  | 			srchedge->bweight = eed->bweight; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* resetting hidden flag */ | 
					
						
							|  |  |  | 	for(numhidden--; numhidden>=0; numhidden--) | 
					
						
							|  |  |  | 		hiddenedges[numhidden]->h= 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* check for orhphan edges */ | 
					
						
							|  |  |  | 	for(srchedge=em->edges.first; srchedge; srchedge=srchedge->next) | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		srchedge->f1= -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* cleanup */ | 
					
						
							|  |  |  | 	MEM_freeN(hiddenedges); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* get rid of the old edge and faces*/ | 
					
						
							|  |  |  | 	remedge(em, eed); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	free_editedge(em, eed); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	BLI_remlink(&em->faces, face[0]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	free_editface(em, face[0]); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	BLI_remlink(&em->faces, face[1]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	free_editface(em, face[1]); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | // XXX ton please check
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* only accepts 1 selected edge, or 2 selected faces */ | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int edge_rotate_selected(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int dir = RNA_int_get(op->ptr, "direction"); // dir == 2 when clockwise and ==1 for counter CW.
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	short edgeCount = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*clear new flag for new edges, count selected edges */ | 
					
						
							|  |  |  | 	for(eed= em->edges.first; eed; eed= eed->next) { | 
					
						
							|  |  |  | 		eed->f1= 0; | 
					
						
							|  |  |  | 		eed->f2 &= ~2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		if(eed->f & SELECT) edgeCount++; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(edgeCount>1) { | 
					
						
							|  |  |  | 		/* more selected edges, check faces */ | 
					
						
							|  |  |  | 		for(efa= em->faces.first; efa; efa= efa->next) { | 
					
						
							|  |  |  | 			if(efa->f & SELECT) { | 
					
						
							|  |  |  | 				efa->e1->f1++; | 
					
						
							|  |  |  | 				efa->e2->f1++; | 
					
						
							|  |  |  | 				efa->e3->f1++; | 
					
						
							|  |  |  | 				if(efa->e4) efa->e4->f1++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		edgeCount= 0; | 
					
						
							|  |  |  | 		for(eed= em->edges.first; eed; eed= eed->next) { | 
					
						
							|  |  |  | 			if(eed->f1==2) edgeCount++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(edgeCount==1) { | 
					
						
							|  |  |  | 			for(eed= em->edges.first; eed; eed= eed->next) { | 
					
						
							|  |  |  | 				if(eed->f1==2) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 					edge_rotate(em, op, eed,dir); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 			BKE_report(op->reports, RPT_ERROR, "Select one edge or two adjacent faces"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 			BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 			return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else if(edgeCount==1) { | 
					
						
							|  |  |  | 		for(eed= em->edges.first; eed; eed= eed->next) { | 
					
						
							|  |  |  | 			if(eed->f & SELECT) { | 
					
						
							|  |  |  | 				EM_select_edge(eed, 0); | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 				edge_rotate(em, op, eed,dir); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	else  { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Select one edge or two adjacent faces"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* flush selected vertices (again) to edges/faces */ | 
					
						
							|  |  |  | 	EM_select_flush(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-29 02:15:13 +00:00
										 |  |  | void MESH_OT_edge_rotate(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Rotate Selected Edge"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Rotate selected edge or adjoining faces."; | 
					
						
							| 
									
										
										
										
											2009-03-29 02:15:13 +00:00
										 |  |  | 	ot->idname= "MESH_OT_edge_rotate"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= edge_rotate_selected; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* props */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	RNA_def_enum(ot->srna, "direction", direction_items, DIRECTION_CW, "direction", "direction to rotate edge around."); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /******************* BEVEL CODE STARTS HERE ********************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* XXX old bevel not ported yet */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | void bevel_menu(EditMesh *em) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BME_Mesh *bm; | 
					
						
							|  |  |  | 	BME_TransData_Head *td; | 
					
						
							|  |  |  | //	TransInfo *t;
 | 
					
						
							|  |  |  | 	int options, res, gbm_free = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //	t = BIF_GetTransInfo();
 | 
					
						
							|  |  |  | 	if (!G.editBMesh) { | 
					
						
							|  |  |  | 		G.editBMesh = MEM_callocN(sizeof(*(G.editBMesh)),"bevel_menu() G.editBMesh"); | 
					
						
							|  |  |  | 		gbm_free = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	G.editBMesh->options = BME_BEVEL_RUNNING | BME_BEVEL_SELECT; | 
					
						
							|  |  |  | 	G.editBMesh->res = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while(G.editBMesh->options & BME_BEVEL_RUNNING) { | 
					
						
							|  |  |  | 		options = G.editBMesh->options; | 
					
						
							|  |  |  | 		res = G.editBMesh->res; | 
					
						
							|  |  |  | 		bm = BME_editmesh_to_bmesh(em); | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | //		BIF_undo_push("Pre-Bevel");
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		free_editMesh(em); | 
					
						
							|  |  |  | 		BME_bevel(bm,0.1f,res,options,0,0,&td); | 
					
						
							| 
									
										
										
										
											2008-12-31 17:11:42 +00:00
										 |  |  | 		BME_bmesh_to_editmesh(bm, td, em); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		EM_selectmode_flush(em); | 
					
						
							|  |  |  | 		G.editBMesh->bm = bm; | 
					
						
							|  |  |  | 		G.editBMesh->td = td; | 
					
						
							|  |  |  | //		initTransform(TFM_BEVEL,CTX_BMESH);
 | 
					
						
							|  |  |  | //		Transform();
 | 
					
						
							|  |  |  | 		BME_free_transdata(td); | 
					
						
							|  |  |  | 		BME_free_mesh(bm); | 
					
						
							|  |  |  | //		if (t->state != TRANS_CONFIRM) {
 | 
					
						
							|  |  |  | //			BIF_undo();
 | 
					
						
							|  |  |  | //		}
 | 
					
						
							|  |  |  | 		if (options == G.editBMesh->options) { | 
					
						
							|  |  |  | 			G.editBMesh->options &= ~BME_BEVEL_RUNNING; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (gbm_free) { | 
					
						
							|  |  |  | 		MEM_freeN(G.editBMesh); | 
					
						
							|  |  |  | 		G.editBMesh = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* *********** END BEVEL *********/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* this utility function checks to see if 2 edit edges share a face,
 | 
					
						
							|  |  |  | returns 1 if they do | 
					
						
							|  |  |  | returns 0 if they do not, or if the function is passed the same edge 2 times | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | short sharesFace(EditMesh *em, EditEdge* e1, EditEdge* e2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *search=NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	search = em->faces.first; | 
					
						
							|  |  |  | 	if (e1 == e2){ | 
					
						
							|  |  |  | 		return 0 ; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	while(search){ | 
					
						
							|  |  |  | 		if( | 
					
						
							|  |  |  | 		   ((search->e1 == e1 || search->e2 == e1) || (search->e3 == e1 || search->e4 == e1)) && | 
					
						
							|  |  |  | 		   ((search->e1 == e2 || search->e2 == e2) || (search->e3 == e2 || search->e4 == e2)) | 
					
						
							|  |  |  | 		   ) { | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		search = search->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SlideUv { | 
					
						
							|  |  |  | 	float origuv[2]; | 
					
						
							|  |  |  | 	float *uv_up, *uv_down; | 
					
						
							|  |  |  | 	//float *fuv[4];
 | 
					
						
							|  |  |  | 	LinkNode *fuv_list; | 
					
						
							|  |  |  | } SlideUv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SlideVert { | 
					
						
							|  |  |  | 	EditEdge *up,*down; | 
					
						
							|  |  |  | 	EditVert origvert; | 
					
						
							|  |  |  | } SlideVert; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-22 08:41:03 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | int EdgeSlide(EditMesh *em, wmOperator *op, short immediate, float imperc) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-20 18:36:27 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | /* XXX REFACTOR - #if 0'd for now, otherwise can't make 64bit windows builds on 64bit machine */ | 
					
						
							|  |  |  | useless: | 
					
						
							|  |  |  | 	goto useless // because it doesn't do anything right now
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 13:30:56 +00:00
										 |  |  | //	NumInput num; XXX
 | 
					
						
							|  |  |  | 	Mesh *me= NULL; // XXX
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	EditEdge *eed,*first=NULL,*last=NULL, *temp = NULL; | 
					
						
							|  |  |  | 	EditVert *ev, *nearest; | 
					
						
							|  |  |  | 	LinkNode *edgelist = NULL, *vertlist=NULL, *look; | 
					
						
							|  |  |  | 	GHash *vertgh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	SlideVert *tempsv; | 
					
						
							| 
									
										
										
										
											2009-05-18 16:42:34 +00:00
										 |  |  | 	float perc = 0, percp = 0,vertdist; // XXX, projectMat[4][4];
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	float shiftlabda= 0.0f,len = 0.0f; | 
					
						
							|  |  |  | 	int i = 0,j, numsel, numadded=0, timesthrough = 0, vertsel=0, prop=1, cancel = 0,flip=0; | 
					
						
							|  |  |  | 	int wasshift = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* UV correction vars */ | 
					
						
							|  |  |  | 	GHash **uvarray= NULL; | 
					
						
							|  |  |  | 	int  uvlay_tot= CustomData_number_of_layers(&em->fdata, CD_MTFACE); | 
					
						
							|  |  |  | 	int uvlay_idx; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	SlideUv *slideuvs=NULL, *suv=NULL, *suv_last=NULL; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	float uv_tmp[2]; | 
					
						
							|  |  |  | 	LinkNode *fuv_link; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	short event, draw=1; | 
					
						
							|  |  |  | 	short mval[2], mvalo[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	char str[128]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	float labda = 0.0f; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | //	initNumInput(&num);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-18 16:42:34 +00:00
										 |  |  | //	view3d_get_object_project_mat(curarea, obedit, projectMat);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	mvalo[0] = -1; mvalo[1] = -1; | 
					
						
							|  |  |  | 	numsel =0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Get number of selected edges and clear some flags
 | 
					
						
							|  |  |  | 	for(eed=em->edges.first;eed;eed=eed->next) { | 
					
						
							|  |  |  | 		eed->f1 = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		eed->f2 = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eed->f & SELECT) numsel++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(ev=em->verts.first;ev;ev=ev->next) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		ev->f1 = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//Make sure each edge only has 2 faces
 | 
					
						
							|  |  |  | 	// make sure loop doesn't cross face
 | 
					
						
							|  |  |  | 	for(efa=em->faces.first;efa;efa=efa->next) { | 
					
						
							|  |  |  | 		int ct = 0; | 
					
						
							|  |  |  | 		if(efa->e1->f & SELECT) { | 
					
						
							|  |  |  | 			ct++; | 
					
						
							|  |  |  | 			efa->e1->f1++; | 
					
						
							|  |  |  | 			if(efa->e1->f1 > 2) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 				BKE_report(op->reports, RPT_ERROR, "3+ face edge"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(efa->e2->f & SELECT) { | 
					
						
							|  |  |  | 			ct++; | 
					
						
							|  |  |  | 			efa->e2->f1++; | 
					
						
							|  |  |  | 			if(efa->e2->f1 > 2) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 				BKE_report(op->reports, RPT_ERROR, "3+ face edge"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(efa->e3->f & SELECT) { | 
					
						
							|  |  |  | 			ct++; | 
					
						
							|  |  |  | 			efa->e3->f1++; | 
					
						
							|  |  |  | 			if(efa->e3->f1 > 2) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 				BKE_report(op->reports, RPT_ERROR, "3+ face edge"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(efa->e4 && efa->e4->f & SELECT) { | 
					
						
							|  |  |  | 			ct++; | 
					
						
							|  |  |  | 			efa->e4->f1++; | 
					
						
							|  |  |  | 			if(efa->e4->f1 > 2) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 				BKE_report(op->reports, RPT_ERROR, "3+ face edge"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		// Make sure loop is not 2 edges of same face
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(ct > 1) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		   BKE_report(op->reports, RPT_ERROR, "Loop crosses itself"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		   return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Get # of selected verts
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	for(ev=em->verts.first;ev;ev=ev->next) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(ev->f & SELECT) vertsel++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Test for multiple segments
 | 
					
						
							|  |  |  | 	if(vertsel > numsel+1) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Please choose a single edge loop"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Get the edgeloop in order - mark f1 with SELECT once added
 | 
					
						
							|  |  |  | 	for(eed=em->edges.first;eed;eed=eed->next) { | 
					
						
							|  |  |  | 		if((eed->f & SELECT) && !(eed->f1 & SELECT)) { | 
					
						
							|  |  |  | 			// If this is the first edge added, just put it in
 | 
					
						
							|  |  |  | 			if(!edgelist) { | 
					
						
							|  |  |  | 				BLI_linklist_prepend(&edgelist,eed); | 
					
						
							|  |  |  | 				numadded++; | 
					
						
							|  |  |  | 				first = eed; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				last  = eed; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				eed->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(editedge_getSharedVert(eed, last)) { | 
					
						
							|  |  |  | 					BLI_linklist_append(&edgelist,eed); | 
					
						
							|  |  |  | 					eed->f1 = SELECT; | 
					
						
							|  |  |  | 					numadded++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					last = eed; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				}  else if(editedge_getSharedVert(eed, first)) { | 
					
						
							|  |  |  | 					BLI_linklist_prepend(&edgelist,eed); | 
					
						
							|  |  |  | 					eed->f1 = SELECT; | 
					
						
							|  |  |  | 					numadded++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					first = eed; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(eed->next == NULL && numadded != numsel) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			eed=em->edges.first; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			timesthrough++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		// It looks like there was an unexpected case - Hopefully should not happen
 | 
					
						
							|  |  |  | 		if(timesthrough >= numsel*2) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			BLI_linklist_free(edgelist,NULL); | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 			BKE_report(op->reports, RPT_ERROR, "Could not order loop"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// Put the verts in order in a linklist
 | 
					
						
							|  |  |  | 	look = edgelist; | 
					
						
							|  |  |  | 	while(look) { | 
					
						
							|  |  |  | 		eed = look->link; | 
					
						
							|  |  |  | 		if(!vertlist) { | 
					
						
							|  |  |  | 			if(look->next) { | 
					
						
							|  |  |  | 				temp = look->next->link; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				//This is the first entry takes care of extra vert
 | 
					
						
							|  |  |  | 				if(eed->v1 != temp->v1 && eed->v1 != temp->v2) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					BLI_linklist_append(&vertlist,eed->v1); | 
					
						
							|  |  |  | 					eed->v1->f1 = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					BLI_linklist_append(&vertlist,eed->v2); | 
					
						
							|  |  |  | 					eed->v2->f1 = 1; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				//This is the case that we only have 1 edge
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				BLI_linklist_append(&vertlist,eed->v1); | 
					
						
							|  |  |  | 				eed->v1->f1 = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		// for all the entries
 | 
					
						
							|  |  |  | 		if(eed->v1->f1 != 1) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			BLI_linklist_append(&vertlist,eed->v1); | 
					
						
							|  |  |  | 			eed->v1->f1 = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} else  if(eed->v2->f1 != 1) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			BLI_linklist_append(&vertlist,eed->v2); | 
					
						
							|  |  |  | 			eed->v2->f1 = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		look = look->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// populate the SlideVerts
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	vertgh = BLI_ghash_new(BLI_ghashutil_ptrhash, BLI_ghashutil_ptrcmp); | 
					
						
							|  |  |  | 	look = vertlist; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	while(look) { | 
					
						
							|  |  |  | 		i=0; | 
					
						
							|  |  |  | 		j=0; | 
					
						
							|  |  |  | 		ev = look->link; | 
					
						
							|  |  |  | 		tempsv = (struct SlideVert*)MEM_mallocN(sizeof(struct SlideVert),"SlideVert"); | 
					
						
							|  |  |  | 		tempsv->up = NULL; | 
					
						
							|  |  |  | 		tempsv->down = NULL; | 
					
						
							|  |  |  | 		tempsv->origvert.co[0] = ev->co[0]; | 
					
						
							|  |  |  | 		tempsv->origvert.co[1] = ev->co[1]; | 
					
						
							|  |  |  | 		tempsv->origvert.co[2] = ev->co[2]; | 
					
						
							|  |  |  | 		tempsv->origvert.no[0] = ev->no[0]; | 
					
						
							|  |  |  | 		tempsv->origvert.no[1] = ev->no[1]; | 
					
						
							|  |  |  | 		tempsv->origvert.no[2] = ev->no[2]; | 
					
						
							|  |  |  | 		// i is total edges that vert is on
 | 
					
						
							|  |  |  | 		// j is total selected edges that vert is on
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(eed=em->edges.first;eed;eed=eed->next) { | 
					
						
							|  |  |  | 			if(eed->v1 == ev || eed->v2 == ev) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				i++; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(eed->f & SELECT) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					 j++; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		// If the vert is in the middle of an edge loop, it touches 2 selected edges and 2 unselected edges
 | 
					
						
							|  |  |  | 		if(i == 4 && j == 2) { | 
					
						
							|  |  |  | 			for(eed=em->edges.first;eed;eed=eed->next) { | 
					
						
							|  |  |  | 				if(editedge_containsVert(eed, ev)) { | 
					
						
							|  |  |  | 					if(!(eed->f & SELECT)) { | 
					
						
							|  |  |  | 						 if(!tempsv->up) { | 
					
						
							|  |  |  | 							 tempsv->up = eed; | 
					
						
							|  |  |  | 						 } else if (!(tempsv->down)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							 tempsv->down = eed; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						 } | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		// If it is on the end of the loop, it touches 1 selected and as least 2 more unselected
 | 
					
						
							|  |  |  | 		if(i >= 3 && j == 1) { | 
					
						
							|  |  |  | 			for(eed=em->edges.first;eed;eed=eed->next) { | 
					
						
							|  |  |  | 				if(editedge_containsVert(eed, ev) && eed->f & SELECT) { | 
					
						
							|  |  |  | 					for(efa = em->faces.first;efa;efa=efa->next) { | 
					
						
							|  |  |  | 						if(editface_containsEdge(efa, eed)) { | 
					
						
							|  |  |  | 							if(editedge_containsVert(efa->e1, ev) && efa->e1 != eed) { | 
					
						
							|  |  |  | 								 if(!tempsv->up) { | 
					
						
							|  |  |  | 									 tempsv->up = efa->e1; | 
					
						
							|  |  |  | 								 } else if (!(tempsv->down)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									 tempsv->down = efa->e1; | 
					
						
							|  |  |  | 								 } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} | 
					
						
							|  |  |  | 							if(editedge_containsVert(efa->e2, ev) && efa->e2 != eed) { | 
					
						
							|  |  |  | 								 if(!tempsv->up) { | 
					
						
							|  |  |  | 									 tempsv->up = efa->e2; | 
					
						
							|  |  |  | 								 } else if (!(tempsv->down)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									 tempsv->down = efa->e2; | 
					
						
							|  |  |  | 								 } | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							if(editedge_containsVert(efa->e3, ev) && efa->e3 != eed) { | 
					
						
							|  |  |  | 								 if(!tempsv->up) { | 
					
						
							|  |  |  | 									 tempsv->up = efa->e3; | 
					
						
							|  |  |  | 								 } else if (!(tempsv->down)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									 tempsv->down = efa->e3; | 
					
						
							|  |  |  | 								 } | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							if(efa->e4) { | 
					
						
							|  |  |  | 								if(editedge_containsVert(efa->e4, ev) && efa->e4 != eed) { | 
					
						
							|  |  |  | 									 if(!tempsv->up) { | 
					
						
							|  |  |  | 										 tempsv->up = efa->e4; | 
					
						
							|  |  |  | 									 } else if (!(tempsv->down)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 										 tempsv->down = efa->e4; | 
					
						
							|  |  |  | 									 } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(i > 4 && j == 2) { | 
					
						
							|  |  |  | 			BLI_ghash_free(vertgh, NULL, (GHashValFreeFP)MEM_freeN); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			BLI_linklist_free(vertlist,NULL); | 
					
						
							|  |  |  | 			BLI_linklist_free(edgelist,NULL); | 
					
						
							|  |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		BLI_ghash_insert(vertgh,ev,tempsv); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		look = look->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// make sure the UPs nad DOWNs are 'faceloops'
 | 
					
						
							|  |  |  | 	// Also find the nearest slidevert to the cursor
 | 
					
						
							|  |  |  | // XXX	getmouseco_areawin(mval);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	look = vertlist; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	nearest = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	vertdist = -1; | 
					
						
							|  |  |  | 	while(look) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		tempsv  = BLI_ghash_lookup(vertgh,(EditVert*)look->link); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if(!tempsv->up || !tempsv->down) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 			BKE_report(op->reports, RPT_ERROR, "Missing rails"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			BLI_ghash_free(vertgh, NULL, (GHashValFreeFP)MEM_freeN); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			BLI_linklist_free(vertlist,NULL); | 
					
						
							|  |  |  | 			BLI_linklist_free(edgelist,NULL); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 13:30:56 +00:00
										 |  |  | 		if(me->drawflag & ME_DRAW_EDGELEN) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(!(tempsv->up->f & SELECT)) { | 
					
						
							|  |  |  | 				tempsv->up->f |= SELECT; | 
					
						
							|  |  |  | 				tempsv->up->f2 |= 16; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				tempsv->up->f2 |= ~16; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(!(tempsv->down->f & SELECT)) { | 
					
						
							|  |  |  | 				tempsv->down->f |= SELECT; | 
					
						
							|  |  |  | 				tempsv->down->f2 |= 16; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				tempsv->down->f2 |= ~16; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(look->next != NULL) { | 
					
						
							|  |  |  | 			SlideVert *sv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			sv = BLI_ghash_lookup(vertgh,(EditVert*)look->next->link); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(sv) { | 
					
						
							|  |  |  | 				float tempdist, co[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(!sharesFace(em, tempsv->up,sv->up)) { | 
					
						
							|  |  |  | 					EditEdge *swap; | 
					
						
							|  |  |  | 					swap = sv->up; | 
					
						
							|  |  |  | 					sv->up = sv->down; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					sv->down = swap; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //				view3d_project_float(curarea, tempsv->origvert.co, co, projectMat);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tempdist = sqrt(pow(co[0] - mval[0],2)+pow(co[1]  - mval[1],2)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(vertdist < 0) { | 
					
						
							|  |  |  | 					vertdist = tempdist; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					nearest  = (EditVert*)look->link; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} else if ( tempdist < vertdist ) { | 
					
						
							|  |  |  | 					vertdist = tempdist; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					nearest  = (EditVert*)look->link; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		look = look->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	if (uvlay_tot) { // XXX && (scene->toolsettings->uvcalc_flag & UVCALC_TRANSFORM_CORRECT)) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		int maxnum = 0; | 
					
						
							|  |  |  | 		uvarray = MEM_callocN( uvlay_tot * sizeof(GHash *), "SlideUVs Array"); | 
					
						
							|  |  |  | 		suv_last = slideuvs = MEM_callocN( uvlay_tot * (numadded+1) * sizeof(SlideUv), "SlideUVs"); /* uvLayers * verts */ | 
					
						
							|  |  |  | 		suv = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for (uvlay_idx=0; uvlay_idx<uvlay_tot; uvlay_idx++) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			uvarray[uvlay_idx] = BLI_ghash_new(BLI_ghashutil_ptrhash, BLI_ghashutil_ptrcmp); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			for(ev=em->verts.first;ev;ev=ev->next) { | 
					
						
							|  |  |  | 				ev->tmp.l = 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			look = vertlist; | 
					
						
							|  |  |  | 			while(look) { | 
					
						
							|  |  |  | 				float *uv_new; | 
					
						
							|  |  |  | 				tempsv  = BLI_ghash_lookup(vertgh,(EditVert*)look->link); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				ev = look->link; | 
					
						
							|  |  |  | 				suv = NULL; | 
					
						
							|  |  |  | 				for(efa = em->faces.first;efa;efa=efa->next) { | 
					
						
							|  |  |  | 					if (ev->tmp.l != -1) { /* test for self, in this case its invalid */ | 
					
						
							|  |  |  | 						int k=-1; /* face corner */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						/* Is this vert in the faces corner? */ | 
					
						
							|  |  |  | 						if		(efa->v1==ev)				k=0; | 
					
						
							|  |  |  | 						else if	(efa->v2==ev)				k=1; | 
					
						
							|  |  |  | 						else if	(efa->v3==ev)				k=2; | 
					
						
							|  |  |  | 						else if	(efa->v4 && efa->v4==ev)	k=3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						if (k != -1) { | 
					
						
							|  |  |  | 							MTFace *tf = CustomData_em_get_n(&em->fdata, efa->data, CD_MTFACE, uvlay_idx); | 
					
						
							|  |  |  | 							EditVert *ev_up, *ev_down; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							uv_new = tf->uv[k]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							if (ev->tmp.l) { | 
					
						
							|  |  |  | 								if (fabs(suv->origuv[0]-uv_new[0]) > 0.0001 || fabs(suv->origuv[1]-uv_new[1])) { | 
					
						
							|  |  |  | 									ev->tmp.l = -1; /* Tag as invalid */ | 
					
						
							|  |  |  | 									BLI_linklist_free(suv->fuv_list,NULL); | 
					
						
							|  |  |  | 									suv->fuv_list = NULL; | 
					
						
							|  |  |  | 									BLI_ghash_remove(uvarray[uvlay_idx],ev, NULL, NULL); | 
					
						
							|  |  |  | 									suv = NULL; | 
					
						
							|  |  |  | 									break; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} else { | 
					
						
							|  |  |  | 								ev->tmp.l = 1; | 
					
						
							|  |  |  | 								suv = suv_last; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								suv->fuv_list = NULL; | 
					
						
							|  |  |  | 								suv->uv_up = suv->uv_down = NULL; | 
					
						
							|  |  |  | 								suv->origuv[0] = uv_new[0]; | 
					
						
							|  |  |  | 								suv->origuv[1] = uv_new[1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								BLI_linklist_prepend(&suv->fuv_list, uv_new); | 
					
						
							|  |  |  | 								BLI_ghash_insert(uvarray[uvlay_idx],ev,suv); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								suv_last++; /* advance to next slide UV */ | 
					
						
							|  |  |  | 								maxnum++; | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							/* Now get the uvs along the up or down edge if we can */ | 
					
						
							|  |  |  | 							if (suv) { | 
					
						
							|  |  |  | 								if (!suv->uv_up) { | 
					
						
							|  |  |  | 									ev_up = editedge_getOtherVert(tempsv->up,ev); | 
					
						
							|  |  |  | 									if		(efa->v1==ev_up)				suv->uv_up = tf->uv[0]; | 
					
						
							|  |  |  | 									else if	(efa->v2==ev_up)				suv->uv_up = tf->uv[1]; | 
					
						
							|  |  |  | 									else if	(efa->v3==ev_up)				suv->uv_up = tf->uv[2]; | 
					
						
							|  |  |  | 									else if	(efa->v4 && efa->v4==ev_up)		suv->uv_up = tf->uv[3]; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 								if (!suv->uv_down) { /* if the first face was apart of the up edge, it cant be apart of the down edge */ | 
					
						
							|  |  |  | 									ev_down = editedge_getOtherVert(tempsv->down,ev); | 
					
						
							|  |  |  | 									if		(efa->v1==ev_down)				suv->uv_down = tf->uv[0]; | 
					
						
							|  |  |  | 									else if	(efa->v2==ev_down)				suv->uv_down = tf->uv[1]; | 
					
						
							|  |  |  | 									else if	(efa->v3==ev_down)				suv->uv_down = tf->uv[2]; | 
					
						
							|  |  |  | 									else if	(efa->v4 && efa->v4==ev_down)	suv->uv_down = tf->uv[3]; | 
					
						
							|  |  |  | 								} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								/* Copy the pointers to the face UV's */ | 
					
						
							|  |  |  | 								BLI_linklist_prepend(&suv->fuv_list, uv_new); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				look = look->next; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} /* end uv layer loop */ | 
					
						
							|  |  |  | 	} /* end uvlay_tot */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	// we should have enough info now to slide
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	len = 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	percp = -1; | 
					
						
							|  |  |  | 	while(draw) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		 /* For the % calculation */ | 
					
						
							|  |  |  | 		short mval[2]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		float rc[2]; | 
					
						
							|  |  |  | 		float v2[2], v3[2]; | 
					
						
							|  |  |  | 		EditVert *centerVert, *upVert, *downVert; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // XXX		getmouseco_areawin(mval);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if (!immediate && (mval[0] == mvalo[0] && mval[1] ==  mvalo[1])) { | 
					
						
							|  |  |  | 			PIL_sleep_ms(10); | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:46:20 +00:00
										 |  |  | 			char *p = str; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			int ctrl= 0, shift= 0; // XXX
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			mvalo[0] = mval[0]; | 
					
						
							|  |  |  | 			mvalo[1] = mval[1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			tempsv = BLI_ghash_lookup(vertgh,nearest); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			centerVert = editedge_getSharedVert(tempsv->up, tempsv->down); | 
					
						
							|  |  |  | 			upVert = editedge_getOtherVert(tempsv->up, centerVert); | 
					
						
							|  |  |  | 			downVert = editedge_getOtherVert(tempsv->down, centerVert); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //			view3d_project_float(curarea, upVert->co, v2, projectMat);
 | 
					
						
							|  |  |  | //			view3d_project_float(curarea, downVert->co, v3, projectMat);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			/* Determine the % on which the loop should be cut */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			rc[0]= v3[0]-v2[0]; | 
					
						
							|  |  |  | 			rc[1]= v3[1]-v2[1]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			len= rc[0]*rc[0]+ rc[1]*rc[1]; | 
					
						
							|  |  |  | 			if (len==0) {len = 0.0001;} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (shift) { | 
					
						
							|  |  |  | 				wasshift = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				labda= ( rc[0]*((mval[0]-v2[0])) + rc[1]*((mval[1]-v2[1])) )/len; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				if (wasshift==0) { | 
					
						
							|  |  |  | 					wasshift = 1; | 
					
						
							|  |  |  | 					shiftlabda = labda; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				labda= ( rc[0]*((mval[0]-v2[0])) + rc[1]*((mval[1]-v2[1])) )/len / 10.0 + shiftlabda; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			if(labda<=0.0) labda=0.0; | 
					
						
							|  |  |  | 			else if(labda>=1.0)labda=1.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			perc=((1-labda)*2)-1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(shift == 0 && ctrl==0) { | 
					
						
							|  |  |  | 				perc *= 100; | 
					
						
							|  |  |  | 				perc = floor(perc); | 
					
						
							|  |  |  | 				perc /= 100; | 
					
						
							|  |  |  | 			} else if (ctrl) { | 
					
						
							|  |  |  | 				perc *= 10; | 
					
						
							|  |  |  | 				perc = floor(perc); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				perc /= 10; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(prop == 0) { | 
					
						
							|  |  |  | 				len = VecLenf(upVert->co,downVert->co)*((perc+1)/2); | 
					
						
							|  |  |  | 				if(flip == 1) { | 
					
						
							|  |  |  | 					len = VecLenf(upVert->co,downVert->co) - len; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if (0) // XXX hasNumInput(&num))
 | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | // XXX				applyNumInput(&num, &perc);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if (prop) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					perc = MIN2(perc, 1); | 
					
						
							|  |  |  | 					perc = MAX2(perc, -1); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					len = MIN2(perc, VecLenf(upVert->co,downVert->co)); | 
					
						
							|  |  |  | 					len = MAX2(len, 0); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			//Adjust Edgeloop
 | 
					
						
							|  |  |  | 			if(immediate) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				perc = imperc; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			percp = perc; | 
					
						
							|  |  |  | 			if(prop) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				look = vertlist; | 
					
						
							|  |  |  | 				while(look) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					EditVert *tempev; | 
					
						
							|  |  |  | 					ev = look->link; | 
					
						
							|  |  |  | 					tempsv = BLI_ghash_lookup(vertgh,ev); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tempev = editedge_getOtherVert((perc>=0)?tempsv->up:tempsv->down, ev); | 
					
						
							|  |  |  | 					VecLerpf(ev->co, tempsv->origvert.co, tempev->co, fabs(perc)); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 					if (0) { // XXX scene->toolsettings->uvcalc_flag & UVCALC_TRANSFORM_CORRECT) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						for (uvlay_idx=0; uvlay_idx<uvlay_tot; uvlay_idx++) { | 
					
						
							|  |  |  | 							suv = BLI_ghash_lookup( uvarray[uvlay_idx], ev ); | 
					
						
							|  |  |  | 							if (suv && suv->fuv_list && suv->uv_up && suv->uv_down) { | 
					
						
							|  |  |  | 								Vec2Lerpf(uv_tmp, suv->origuv,  (perc>=0)?suv->uv_up:suv->uv_down, fabs(perc)); | 
					
						
							|  |  |  | 								fuv_link = suv->fuv_list; | 
					
						
							|  |  |  | 								while (fuv_link) { | 
					
						
							|  |  |  | 									VECCOPY2D(((float *)fuv_link->link), uv_tmp); | 
					
						
							|  |  |  | 									fuv_link = fuv_link->next; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					look = look->next; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				//Non prop code
 | 
					
						
							|  |  |  | 				look = vertlist; | 
					
						
							|  |  |  | 				while(look) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					float newlen; | 
					
						
							|  |  |  | 					ev = look->link; | 
					
						
							|  |  |  | 					tempsv = BLI_ghash_lookup(vertgh,ev); | 
					
						
							|  |  |  | 					newlen = (len / VecLenf(editedge_getOtherVert(tempsv->up,ev)->co,editedge_getOtherVert(tempsv->down,ev)->co)); | 
					
						
							|  |  |  | 					if(newlen > 1.0) {newlen = 1.0;} | 
					
						
							|  |  |  | 					if(newlen < 0.0) {newlen = 0.0;} | 
					
						
							|  |  |  | 					if(flip == 0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						VecLerpf(ev->co, editedge_getOtherVert(tempsv->down,ev)->co, editedge_getOtherVert(tempsv->up,ev)->co, fabs(newlen)); | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 						if (0) { // XXX scene->toolsettings->uvcalc_flag & UVCALC_TRANSFORM_CORRECT) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							/* dont do anything if no UVs */ | 
					
						
							|  |  |  | 							for (uvlay_idx=0; uvlay_idx<uvlay_tot; uvlay_idx++) { | 
					
						
							|  |  |  | 								suv = BLI_ghash_lookup( uvarray[uvlay_idx], ev ); | 
					
						
							|  |  |  | 								if (suv && suv->fuv_list && suv->uv_up && suv->uv_down) { | 
					
						
							|  |  |  | 									Vec2Lerpf(uv_tmp, suv->uv_down, suv->uv_up, fabs(newlen)); | 
					
						
							|  |  |  | 									fuv_link = suv->fuv_list; | 
					
						
							|  |  |  | 									while (fuv_link) { | 
					
						
							|  |  |  | 										VECCOPY2D(((float *)fuv_link->link), uv_tmp); | 
					
						
							|  |  |  | 										fuv_link = fuv_link->next; | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} else{ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						VecLerpf(ev->co, editedge_getOtherVert(tempsv->up,ev)->co, editedge_getOtherVert(tempsv->down,ev)->co, fabs(newlen)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 						if (0) { // XXX scene->toolsettings->uvcalc_flag & UVCALC_TRANSFORM_CORRECT) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							/* dont do anything if no UVs */ | 
					
						
							|  |  |  | 							for (uvlay_idx=0; uvlay_idx<uvlay_tot; uvlay_idx++) { | 
					
						
							|  |  |  | 								suv = BLI_ghash_lookup( uvarray[uvlay_idx], ev ); | 
					
						
							|  |  |  | 								if (suv && suv->fuv_list && suv->uv_up && suv->uv_down) { | 
					
						
							|  |  |  | 									Vec2Lerpf(uv_tmp, suv->uv_up, suv->uv_down, fabs(newlen)); | 
					
						
							|  |  |  | 									fuv_link = suv->fuv_list; | 
					
						
							|  |  |  | 									while (fuv_link) { | 
					
						
							|  |  |  | 										VECCOPY2D(((float *)fuv_link->link), uv_tmp); | 
					
						
							|  |  |  | 										fuv_link = fuv_link->next; | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					look = look->next; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			 // Highlight the Control Edges
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | //			scrarea_do_windraw(curarea);
 | 
					
						
							|  |  |  | //			persp(PERSP_VIEW);
 | 
					
						
							|  |  |  | //			glPushMatrix();
 | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | //			mymultmatrix(obedit->obmat);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			glColor3ub(0, 255, 0); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			glBegin(GL_LINES); | 
					
						
							|  |  |  | 			glVertex3fv(upVert->co); | 
					
						
							|  |  |  | 			glVertex3fv(downVert->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			glEnd(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(prop == 0) { | 
					
						
							|  |  |  | 				// draw start edge for non-prop
 | 
					
						
							|  |  |  | 				glPointSize(5); | 
					
						
							|  |  |  | 				glBegin(GL_POINTS); | 
					
						
							|  |  |  | 				glColor3ub(255,0,255); | 
					
						
							|  |  |  | 				if(flip) { | 
					
						
							|  |  |  | 					glVertex3fv(upVert->co); | 
					
						
							|  |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					glVertex3fv(downVert->co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				glEnd(); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			glPopMatrix(); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if(prop) { | 
					
						
							|  |  |  | 				p += sprintf(str, "(P)ercentage: "); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				p += sprintf(str, "Non (P)rop Length: "); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if (0) // XXX hasNumInput(&num))
 | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				char num_str[20]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				// XX outputNumInput(&num, num_str);
 | 
					
						
							|  |  |  | 				p += sprintf(p, "%s", num_str); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				if (prop) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					p += sprintf(p, "%f", perc); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					p += sprintf(p, "%f", len); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if (prop == 0) { | 
					
						
							|  |  |  | 				p += sprintf(p, ", Press (F) to flip control side"); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //			headerprint(str);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | //			screen_swapbuffers();
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		if(!immediate) { | 
					
						
							|  |  |  | 			while(qtest()) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				short val=0; | 
					
						
							|  |  |  | 				event= extern_qread(&val);	// extern_qread stores important events for the mainloop to handle
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				/* val==0 on key-release event */ | 
					
						
							|  |  |  | 				if (val) { | 
					
						
							|  |  |  | 					if(ELEM(event, ESCKEY, RIGHTMOUSE)) { | 
					
						
							|  |  |  | 							prop = 1; // Go back to prop mode
 | 
					
						
							|  |  |  | 							imperc = 0; // This is the % that gets set for immediate
 | 
					
						
							|  |  |  | 							immediate = 1; //Run through eval code 1 more time
 | 
					
						
							|  |  |  | 							cancel = 1;   // Return -1
 | 
					
						
							|  |  |  | 							mvalo[0] = -1; | 
					
						
							|  |  |  | 					} else if(ELEM3(event, PADENTER, LEFTMOUSE, RETKEY)) { | 
					
						
							|  |  |  | 							draw = 0; // End looping now
 | 
					
						
							|  |  |  | 					} else if(event==MIDDLEMOUSE) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							perc = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							immediate = 1; | 
					
						
							|  |  |  | 					} else if(event==PKEY) { | 
					
						
							|  |  |  | // XXX							initNumInput(&num); /* reset num input */
 | 
					
						
							|  |  |  | 							if (prop) { | 
					
						
							|  |  |  | 								prop = 0; | 
					
						
							|  |  |  | // XXX								num.flag |= NUM_NO_NEGATIVE;
 | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								prop = 1; | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							mvalo[0] = -1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} else if(event==FKEY) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							(flip == 1) ? (flip = 0):(flip = 1); | 
					
						
							|  |  |  | 							mvalo[0] = -1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} else if(ELEM(event, RIGHTARROWKEY, WHEELUPMOUSE)) { // Scroll through Control Edges
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						look = vertlist; | 
					
						
							|  |  |  | 				 		while(look) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							if(nearest == (EditVert*)look->link) { | 
					
						
							|  |  |  | 								if(look->next == NULL) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									nearest =  (EditVert*)vertlist->link; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								} else { | 
					
						
							|  |  |  | 									nearest = (EditVert*)look->next->link; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 								} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 								mvalo[0] = -1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 								break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 							look = look->next; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} else if(ELEM(event, LEFTARROWKEY, WHEELDOWNMOUSE)) { // Scroll through Control Edges
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						look = vertlist; | 
					
						
							|  |  |  | 				 		while(look) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							if(look->next) { | 
					
						
							|  |  |  | 								if(look->next->link == nearest) { | 
					
						
							|  |  |  | 									nearest = (EditVert*)look->link; | 
					
						
							|  |  |  | 									mvalo[0] = -1; | 
					
						
							|  |  |  | 									break; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 								} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							} else { | 
					
						
							|  |  |  | 								if((EditVert*)vertlist->link == nearest) { | 
					
						
							|  |  |  | 									nearest = look->link; | 
					
						
							|  |  |  | 									mvalo[0] = -1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									break; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							look = look->next; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | // XXX					if (handleNumInput(&num, event))
 | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						mvalo[0] = -1; /* NEED A BETTER WAY TO TRIGGER REDRAW */ | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			draw = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | //		DAG_id_flush_update(obedit->data, OB_RECALC_DATA);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 13:30:56 +00:00
										 |  |  | 	if(me->drawflag & ME_DRAW_EDGELEN) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		look = vertlist; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		while(look) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			tempsv  = BLI_ghash_lookup(vertgh,(EditVert*)look->link); | 
					
						
							|  |  |  | 			if(tempsv != NULL) { | 
					
						
							|  |  |  | 				tempsv->up->f &= !SELECT; | 
					
						
							|  |  |  | 				tempsv->down->f &= !SELECT; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			look = look->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | //	force_draw(0);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!immediate) | 
					
						
							|  |  |  | 		EM_automerge(0); | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | //	DAG_id_flush_update(obedit->data, OB_RECALC_DATA);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | //	scrarea_queue_winredraw(curarea);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//BLI_ghash_free(edgesgh, freeGHash, NULL);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	BLI_ghash_free(vertgh, NULL, (GHashValFreeFP)MEM_freeN); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	BLI_linklist_free(vertlist,NULL); | 
					
						
							|  |  |  | 	BLI_linklist_free(edgelist,NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	if (uvlay_tot) { // XXX  && (scene->toolsettings->uvcalc_flag & UVCALC_TRANSFORM_CORRECT)) {
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for (uvlay_idx=0; uvlay_idx<uvlay_tot; uvlay_idx++) { | 
					
						
							|  |  |  | 			BLI_ghash_free(uvarray[uvlay_idx], NULL, NULL); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		MEM_freeN(uvarray); | 
					
						
							|  |  |  | 		MEM_freeN(slideuvs); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		suv = suv_last-1; | 
					
						
							|  |  |  | 		while (suv >= slideuvs) { | 
					
						
							|  |  |  | 			if (suv->fuv_list) { | 
					
						
							|  |  |  | 				BLI_linklist_free(suv->fuv_list,NULL); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			suv--; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(cancel == 1) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-09-22 08:41:03 +00:00
										 |  |  | #endif // END OF XXX
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | int EdgeLoopDelete(EditMesh *em, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-09-22 08:41:03 +00:00
										 |  |  | #if 0 //XXX won't work with new edgeslide
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* temporal flag setting so we keep UVs when deleting edge loops,
 | 
					
						
							|  |  |  | 	* this is a bit of a hack but it works how you would want in almost all cases */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	//	short uvcalc_flag_orig = 0; // XXX scene->toolsettings->uvcalc_flag;
 | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	//	scene->toolsettings->uvcalc_flag |= UVCALC_TRANSFORM_CORRECT;
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 	if(!EdgeSlide(em, op, 1, 1)) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* restore uvcalc flag */ | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | 	//	scene->toolsettings->uvcalc_flag = uvcalc_flag_orig;
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EM_select_more(em); | 
					
						
							|  |  |  | 	removedoublesflag(em, 1,0, 0.001); | 
					
						
							|  |  |  | 	EM_select_flush(em); | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	//	DAG_id_flush_update(obedit->data, OB_RECALC_DATA);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return 1; | 
					
						
							| 
									
										
										
										
											2009-09-22 08:41:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-09-22 17:50:29 +00:00
										 |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------- More tools ------------------ */ | 
					
						
							| 
									
										
										
										
											2009-02-16 14:28:33 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | void mesh_set_face_flags(EditMesh *em, short mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	MTFace *tface; | 
					
						
							| 
									
										
										
										
											2009-06-16 01:22:56 +00:00
										 |  |  | 	short	m_tex=0, m_shared=0, | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			m_light=0, m_invis=0, m_collision=0, | 
					
						
							|  |  |  | 			m_twoside=0, m_obcolor=0, m_halo=0, | 
					
						
							|  |  |  | 			m_billboard=0, m_shadow=0, m_text=0, | 
					
						
							|  |  |  | 			m_sort=0; | 
					
						
							|  |  |  | 	short flag = 0, change = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | // XXX	if (!EM_texFaceCheck()) {
 | 
					
						
							|  |  |  | //		error("not a mesh with uv/image layers");
 | 
					
						
							|  |  |  | //		return;
 | 
					
						
							|  |  |  | //	}
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	add_numbut(0, TOG|SHO, "Texture", 0, 0, &m_tex, NULL); | 
					
						
							|  |  |  | 	add_numbut(2, TOG|SHO, "Light", 0, 0, &m_light, NULL); | 
					
						
							|  |  |  | 	add_numbut(3, TOG|SHO, "Invisible", 0, 0, &m_invis, NULL); | 
					
						
							|  |  |  | 	add_numbut(4, TOG|SHO, "Collision", 0, 0, &m_collision, NULL); | 
					
						
							|  |  |  | 	add_numbut(5, TOG|SHO, "Shared", 0, 0, &m_shared, NULL); | 
					
						
							|  |  |  | 	add_numbut(6, TOG|SHO, "Twoside", 0, 0, &m_twoside, NULL); | 
					
						
							|  |  |  | 	add_numbut(7, TOG|SHO, "ObColor", 0, 0, &m_obcolor, NULL); | 
					
						
							|  |  |  | 	add_numbut(8, TOG|SHO, "Halo", 0, 0, &m_halo, NULL); | 
					
						
							|  |  |  | 	add_numbut(9, TOG|SHO, "Billboard", 0, 0, &m_billboard, NULL); | 
					
						
							|  |  |  | 	add_numbut(10, TOG|SHO, "Shadow", 0, 0, &m_shadow, NULL); | 
					
						
							|  |  |  | 	add_numbut(11, TOG|SHO, "Text", 0, 0, &m_text, NULL); | 
					
						
							|  |  |  | 	add_numbut(12, TOG|SHO, "Sort", 0, 0, &m_sort, NULL); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (!do_clever_numbuts((mode ? "Set Flags" : "Clear Flags"), 13, REDRAW)) | 
					
						
							|  |  |  |  		return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* these 2 cant both be on */ | 
					
						
							|  |  |  | 	if (mode) /* are we seeting*/ | 
					
						
							|  |  |  | 		if (m_halo) | 
					
						
							|  |  |  | 			m_billboard = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (m_tex)			flag |= TF_TEX; | 
					
						
							|  |  |  | 	if (m_shared)		flag |= TF_SHAREDCOL; | 
					
						
							|  |  |  | 	if (m_light)		flag |= TF_LIGHT; | 
					
						
							|  |  |  | 	if (m_invis)		flag |= TF_INVISIBLE; | 
					
						
							|  |  |  | 	if (m_collision)	flag |= TF_DYNAMIC; | 
					
						
							|  |  |  | 	if (m_twoside)		flag |= TF_TWOSIDE; | 
					
						
							|  |  |  | 	if (m_obcolor)		flag |= TF_OBCOL; | 
					
						
							|  |  |  | 	if (m_halo)			flag |= TF_BILLBOARD; | 
					
						
							|  |  |  | 	if (m_billboard)	flag |= TF_BILLBOARD2; | 
					
						
							|  |  |  | 	if (m_shadow)		flag |= TF_SHADOW; | 
					
						
							|  |  |  | 	if (m_text)			flag |= TF_BMFONT; | 
					
						
							|  |  |  | 	if (m_sort)			flag |= TF_ALPHASORT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (flag==0) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	efa= em->faces.first; | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		if(efa->f & SELECT) { | 
					
						
							|  |  |  | 			tface= CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							|  |  |  | 			if (mode)	tface->mode |= flag; | 
					
						
							|  |  |  | 			else		tface->mode &= ~flag; | 
					
						
							|  |  |  | 			change = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		efa= efa->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-02-16 14:28:33 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** Rip Operator *************************/ | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* helper to find edge for edge_rip */ | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | static float mesh_rip_edgedist(ARegion *ar, float mat[][4], float *co1, float *co2, short *mval) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec1[3], vec2[3], mvalf[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	view3d_project_float(ar, co1, vec1, mat); | 
					
						
							|  |  |  | 	view3d_project_float(ar, co2, vec2, mat); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	mvalf[0]= (float)mval[0]; | 
					
						
							|  |  |  | 	mvalf[1]= (float)mval[1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return PdistVL2Dfl(mvalf, vec1, vec2); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* helper for below */ | 
					
						
							|  |  |  | static void mesh_rip_setface(EditMesh *em, EditFace *sefa) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* put new vertices & edges in best face */ | 
					
						
							|  |  |  | 	if(sefa->v1->tmp.v) sefa->v1= sefa->v1->tmp.v; | 
					
						
							|  |  |  | 	if(sefa->v2->tmp.v) sefa->v2= sefa->v2->tmp.v; | 
					
						
							|  |  |  | 	if(sefa->v3->tmp.v) sefa->v3= sefa->v3->tmp.v; | 
					
						
							|  |  |  | 	if(sefa->v4 && sefa->v4->tmp.v) sefa->v4= sefa->v4->tmp.v; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	sefa->e1= addedgelist(em, sefa->v1, sefa->v2, sefa->e1); | 
					
						
							|  |  |  | 	sefa->e2= addedgelist(em, sefa->v2, sefa->v3, sefa->e2); | 
					
						
							|  |  |  | 	if(sefa->v4) { | 
					
						
							|  |  |  | 		sefa->e3= addedgelist(em, sefa->v3, sefa->v4, sefa->e3); | 
					
						
							|  |  |  | 		sefa->e4= addedgelist(em, sefa->v4, sefa->v1, sefa->e4); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		sefa->e3= addedgelist(em, sefa->v3, sefa->v1, sefa->e3); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* based on mouse cursor position, it defines how is being ripped */ | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | static int mesh_rip_invoke(bContext *C, wmOperator *op, wmEvent *event) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	ARegion *ar= CTX_wm_region(C); | 
					
						
							|  |  |  | 	RegionView3D *rv3d= ar->regiondata; | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditVert *eve, *nextve; | 
					
						
							|  |  |  | 	EditEdge *eed, *seed= NULL; | 
					
						
							|  |  |  | 	EditFace *efa, *sefa= NULL; | 
					
						
							| 
									
										
										
										
											2009-05-18 16:42:34 +00:00
										 |  |  | 	float projectMat[4][4], vec[3], dist, mindist; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	short doit= 1, *mval= event->mval; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* select flush... vertices are important */ | 
					
						
							|  |  |  | 	EM_selectmode_set(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-18 16:42:34 +00:00
										 |  |  | 	view3d_get_object_project_mat(rv3d, obedit, projectMat); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* find best face, exclude triangles and break on face select or faces with 2 edges select */ | 
					
						
							|  |  |  | 	mindist= 1000000.0f; | 
					
						
							|  |  |  | 	for(efa= em->faces.first; efa; efa=efa->next) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		if( efa->f & 1) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		if(efa->v4 && faceselectedOR(efa, SELECT) ) { | 
					
						
							|  |  |  | 			int totsel=0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(efa->e1->f & SELECT) totsel++; | 
					
						
							|  |  |  | 			if(efa->e2->f & SELECT) totsel++; | 
					
						
							|  |  |  | 			if(efa->e3->f & SELECT) totsel++; | 
					
						
							|  |  |  | 			if(efa->e4->f & SELECT) totsel++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(totsel>1) | 
					
						
							|  |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			view3d_project_float(ar, efa->cent, vec, projectMat); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			dist= sqrt( (vec[0]-mval[0])*(vec[0]-mval[0]) + (vec[1]-mval[1])*(vec[1]-mval[1]) ); | 
					
						
							|  |  |  | 			if(dist<mindist) { | 
					
						
							|  |  |  | 				mindist= dist; | 
					
						
							|  |  |  | 				sefa= efa; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(efa) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Can't perform ripping with faces selected this way"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if(sefa==NULL) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "No proper selection or faces included"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* duplicate vertices, new vertices get selected */ | 
					
						
							|  |  |  | 	for(eve = em->verts.last; eve; eve= eve->prev) { | 
					
						
							|  |  |  | 		eve->tmp.v = NULL; | 
					
						
							|  |  |  | 		if(eve->f & SELECT) { | 
					
						
							|  |  |  | 			eve->tmp.v = addvertlist(em, eve->co, eve); | 
					
						
							|  |  |  | 			eve->f &= ~SELECT; | 
					
						
							|  |  |  | 			eve->tmp.v->f |= SELECT; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* find the best candidate edge */ | 
					
						
							|  |  |  | 	/* or one of sefa edges is selected... */ | 
					
						
							|  |  |  | 	if(sefa->e1->f & SELECT) seed= sefa->e2; | 
					
						
							|  |  |  | 	if(sefa->e2->f & SELECT) seed= sefa->e1; | 
					
						
							|  |  |  | 	if(sefa->e3->f & SELECT) seed= sefa->e2; | 
					
						
							|  |  |  | 	if(sefa->e4 && sefa->e4->f & SELECT) seed= sefa->e3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* or we do the distance trick */ | 
					
						
							|  |  |  | 	if(seed==NULL) { | 
					
						
							|  |  |  | 		mindist= 1000000.0f; | 
					
						
							|  |  |  | 		if(sefa->e1->v1->tmp.v || sefa->e1->v2->tmp.v) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			dist = mesh_rip_edgedist(ar, projectMat, | 
					
						
							|  |  |  | 									 sefa->e1->v1->co, | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 									 sefa->e1->v2->co, mval); | 
					
						
							|  |  |  | 			if(dist<mindist) { | 
					
						
							|  |  |  | 				seed= sefa->e1; | 
					
						
							|  |  |  | 				mindist= dist; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(sefa->e2->v1->tmp.v || sefa->e2->v2->tmp.v) { | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			dist = mesh_rip_edgedist(ar, projectMat, | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 									 sefa->e2->v1->co, | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 									 sefa->e2->v2->co, mval); | 
					
						
							|  |  |  | 			if(dist<mindist) { | 
					
						
							|  |  |  | 				seed= sefa->e2; | 
					
						
							|  |  |  | 				mindist= dist; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(sefa->e3->v1->tmp.v || sefa->e3->v2->tmp.v) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			dist= mesh_rip_edgedist(ar, projectMat, | 
					
						
							|  |  |  | 									sefa->e3->v1->co, | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 									sefa->e3->v2->co, mval); | 
					
						
							|  |  |  | 			if(dist<mindist) { | 
					
						
							|  |  |  | 				seed= sefa->e3; | 
					
						
							|  |  |  | 				mindist= dist; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(sefa->e4 && (sefa->e4->v1->tmp.v || sefa->e4->v2->tmp.v)) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			dist= mesh_rip_edgedist(ar, projectMat, | 
					
						
							|  |  |  | 									sefa->e4->v1->co, | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 									sefa->e4->v2->co, mval); | 
					
						
							|  |  |  | 			if(dist<mindist) { | 
					
						
							|  |  |  | 				seed= sefa->e4; | 
					
						
							|  |  |  | 				mindist= dist; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(seed==NULL) {	// never happens?
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "No proper edge found to start"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	faceloop_select(em, seed, 2);	// tmp abuse for finding all edges that need duplicated, returns OK faces with f1
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* duplicate edges in the loop, with at least 1 vertex selected, needed for selection flip */ | 
					
						
							|  |  |  | 	for(eed = em->edges.last; eed; eed= eed->prev) { | 
					
						
							|  |  |  | 		eed->tmp.v = NULL; | 
					
						
							|  |  |  | 		if((eed->v1->tmp.v) || (eed->v2->tmp.v)) { | 
					
						
							|  |  |  | 			EditEdge *newed; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			newed= addedgelist(em, eed->v1->tmp.v?eed->v1->tmp.v:eed->v1, | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							   eed->v2->tmp.v?eed->v2->tmp.v:eed->v2, eed); | 
					
						
							|  |  |  | 			if(eed->f & SELECT) { | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 				EM_select_edge(eed, 0); | 
					
						
							|  |  |  | 				EM_remove_selection(em, eed, EDITEDGE); | 
					
						
							|  |  |  | 				EM_select_edge(newed, 1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			eed->tmp.v = (EditVert *)newed; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* first clear edges to help finding neighbours */ | 
					
						
							|  |  |  | 	for(eed = em->edges.last; eed; eed= eed->prev) eed->f1= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* put new vertices & edges && flag in best face */ | 
					
						
							|  |  |  | 	mesh_rip_setface(em, sefa); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* starting with neighbours of best face, we loop over the seam */ | 
					
						
							|  |  |  | 	sefa->f1= 2; | 
					
						
							|  |  |  | 	doit= 1; | 
					
						
							|  |  |  | 	while(doit) { | 
					
						
							|  |  |  | 		doit= 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(efa= em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 			/* new vert in face */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			if (efa->v1->tmp.v || efa->v2->tmp.v || | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				efa->v3->tmp.v || (efa->v4 && efa->v4->tmp.v)) { | 
					
						
							|  |  |  | 				/* face is tagged with loop */ | 
					
						
							|  |  |  | 				if(efa->f1==1) { | 
					
						
							|  |  |  | 					mesh_rip_setface(em, efa); | 
					
						
							|  |  |  | 					efa->f1= 2; | 
					
						
							|  |  |  | 					doit= 1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* remove loose edges, that were part of a ripped face */ | 
					
						
							|  |  |  | 	for(eve = em->verts.first; eve; eve= eve->next) eve->f1= 0; | 
					
						
							|  |  |  | 	for(eed = em->edges.last; eed; eed= eed->prev) eed->f1= 0; | 
					
						
							|  |  |  | 	for(efa= em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 		efa->e1->f1= 1; | 
					
						
							|  |  |  | 		efa->e2->f1= 1; | 
					
						
							|  |  |  | 		efa->e3->f1= 1; | 
					
						
							|  |  |  | 		if(efa->e4) efa->e4->f1= 1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(eed = em->edges.last; eed; eed= seed) { | 
					
						
							|  |  |  | 		seed= eed->prev; | 
					
						
							|  |  |  | 		if(eed->f1==0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			if(eed->v1->tmp.v || eed->v2->tmp.v || | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			   (eed->v1->f & SELECT) || (eed->v2->f & SELECT)) { | 
					
						
							|  |  |  | 				remedge(em, eed); | 
					
						
							|  |  |  | 				free_editedge(em, eed); | 
					
						
							|  |  |  | 				eed= NULL; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(eed) { | 
					
						
							|  |  |  | 			eed->v1->f1= 1; | 
					
						
							|  |  |  | 			eed->v2->f1= 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* and remove loose selected vertices, that got duplicated accidentally */ | 
					
						
							|  |  |  | 	for(eve = em->verts.first; eve; eve= nextve) { | 
					
						
							|  |  |  | 		nextve= eve->next; | 
					
						
							|  |  |  | 		if(eve->f1==0 && (eve->tmp.v || (eve->f & SELECT))) { | 
					
						
							|  |  |  | 			BLI_remlink(&em->verts,eve); | 
					
						
							|  |  |  | 			free_editvert(em, eve); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Operator goodies!
--- Macro operators
Operators now can consist of multiple operators. Such a macro operator
is identical and behaves identical to other opererators. Macros can
also be constructed of macros even! Currently only hardcoded macros are
implemented, this to solve combined operators such as 'add duplicate' or
'extrude' (both want a transform appended).
Usage is simple:
- WM_operatortype_append_macro() : add new operatortype, name, flags
- WM_operatortype_macro_define() : add existing operator to macro
(Note: macro_define will also allow properties to be set, doesnt work
 right now)
On converting the macro wmOperatorType to a real operator, it makes a
list of all operators, and the standard macro callbacks (exec, invoke,
modal, poll) just will use all.
Important note; switching to a modal operator only works as last in the
chain now!
Macros implemented for duplicate, extrude and rip. Tool menu works fine
for it, also the redo hotkey F4 works properly.
--- Operator redo fix
The operators use the undo system to switch back, but this could give
errors if other actions added undo pushes (buttons, outliner). Now the
redo for operator searches back for the correct undo level.
This fixes issues with many redos.
Note for brecht: removed the ED_undo_push for buttons... it was called
on *every* button now, which is probably too much? For example, using
the 'toolbar' redo also caused this...
											
										 
											2009-07-29 17:56:38 +00:00
										 |  |  | //	RNA_enum_set(op->ptr, "proportional", 0);
 | 
					
						
							|  |  |  | //	RNA_boolean_set(op->ptr, "mirror", 0);
 | 
					
						
							|  |  |  | //	WM_operator_name_call(C, "TFM_OT_translate", WM_OP_INVOKE_REGION_WIN, op->ptr);
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | void MESH_OT_rip(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Rip"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Rip selection from mesh (quads only)."; | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	ot->idname= "MESH_OT_rip"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->invoke= mesh_rip_invoke; | 
					
						
							| 
									
										
										
										
											2009-06-16 13:09:36 +00:00
										 |  |  | 	ot->poll= EM_view3d_poll; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	/* to give to transform */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	Properties_Proportional(ot); | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "mirror", 0, "Mirror Editing", ""); | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /************************ Shape Operators *************************/ | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | void shape_propagate(Scene *scene, Object *obedit, EditMesh *em, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert *ev = NULL; | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	Mesh* me = (Mesh*)obedit->data; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Key*  ky = NULL; | 
					
						
							|  |  |  | 	KeyBlock* kb = NULL; | 
					
						
							|  |  |  | 	Base* base=NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(me->key){ | 
					
						
							|  |  |  | 		ky = me->key; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Object Has No Key"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(ky->block.first){ | 
					
						
							|  |  |  | 		for(ev = em->verts.first; ev ; ev = ev->next){ | 
					
						
							|  |  |  | 			if(ev->f & SELECT){ | 
					
						
							|  |  |  | 				for(kb=ky->block.first;kb;kb = kb->next){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					float *data; | 
					
						
							|  |  |  | 					data = kb->data; | 
					
						
							|  |  |  | 					VECCOPY(data+(ev->keyindex*3),ev->co); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Object Has No Blendshapes"); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//TAG Mesh Objects that share this data
 | 
					
						
							|  |  |  | 	for(base = scene->base.first; base; base = base->next){ | 
					
						
							|  |  |  | 		if(base->object && base->object->data == me){ | 
					
						
							|  |  |  | 			base->object->recalc = OB_RECALC_DATA; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void shape_copy_from_lerp(EditMesh *em, KeyBlock* thisBlock, KeyBlock* fromBlock) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *ev = NULL; | 
					
						
							|  |  |  | 	short mval[2], curval[2], event = 0, finished = 0, canceled = 0, fullcopy=0 ; | 
					
						
							|  |  |  | 	float perc = 0; | 
					
						
							|  |  |  | 	char str[64]; | 
					
						
							|  |  |  | 	float *data, *odata; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	data  = fromBlock->data; | 
					
						
							|  |  |  | 	odata = thisBlock->data; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | // XXX	getmouseco_areawin(mval);
 | 
					
						
							|  |  |  | 	curval[0] = mval[0] + 1; curval[1] = mval[1] + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (finished == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | // XXX		getmouseco_areawin(mval);
 | 
					
						
							|  |  |  | 		if (mval[0] != curval[0] || mval[1] != curval[1]) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(mval[0] > curval[0]) | 
					
						
							|  |  |  | 				perc += 0.1; | 
					
						
							|  |  |  | 			else if(mval[0] < curval[0]) | 
					
						
							|  |  |  | 				perc -= 0.1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(perc < 0) perc = 0; | 
					
						
							|  |  |  | 			if(perc > 1) perc = 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			curval[0] = mval[0]; | 
					
						
							|  |  |  | 			curval[1] = mval[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(fullcopy == 1){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				perc = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for(ev = em->verts.first; ev ; ev = ev->next){ | 
					
						
							|  |  |  | 				if(ev->f & SELECT){ | 
					
						
							|  |  |  | 					VecLerpf(ev->co,odata+(ev->keyindex*3),data+(ev->keyindex*3),perc); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			sprintf(str,"Blending at %d%c  MMB to Copy at 100%c",(int)(perc*100),'%','%'); | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | //			DAG_id_flush_update(obedit->data, OB_RECALC_DATA);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | //			headerprint(str);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | //			force_draw(0);
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if(fullcopy == 1){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			PIL_sleep_ms(10); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		while(qtest()) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			short val=0; | 
					
						
							|  |  |  | 			event= extern_qread(&val); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(val){ | 
					
						
							|  |  |  | 				if(ELEM3(event, PADENTER, LEFTMOUSE, RETKEY)){ | 
					
						
							|  |  |  | 					finished = 1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else if (event == MIDDLEMOUSE){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					fullcopy = 1; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else if (ELEM3(event,ESCKEY,RIGHTMOUSE,RIGHTMOUSE)){ | 
					
						
							|  |  |  | 					canceled = 1; | 
					
						
							|  |  |  | 					finished = 1; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	if(!canceled); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		for(ev = em->verts.first; ev ; ev = ev->next){ | 
					
						
							|  |  |  | 			if(ev->f & SELECT){ | 
					
						
							|  |  |  | 				VECCOPY(ev->co, odata+(ev->keyindex*3)); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | void shape_copy_select_from(Object *obedit, EditMesh *em, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	Mesh* me = (Mesh*)obedit->data; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditVert *ev = NULL; | 
					
						
							| 
									
										
										
										
											2009-01-02 19:10:35 +00:00
										 |  |  | 	int totverts = 0,curshape = obedit->shapenr; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Key*  ky = NULL; | 
					
						
							|  |  |  | 	KeyBlock *kb = NULL,*thisBlock = NULL; | 
					
						
							|  |  |  | 	int maxlen=32, nr=0, a=0; | 
					
						
							|  |  |  | 	char *menu; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(me->key){ | 
					
						
							|  |  |  | 		ky = me->key; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Object Has No Key"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(ky->block.first){ | 
					
						
							|  |  |  | 		for(kb=ky->block.first;kb;kb = kb->next){ | 
					
						
							|  |  |  | 			maxlen += 40; // Size of a block name
 | 
					
						
							|  |  |  | 			if(a == curshape-1){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					thisBlock = kb; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			a++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		a=0; | 
					
						
							|  |  |  | 		menu = MEM_callocN(maxlen, "Copy Shape Menu Text"); | 
					
						
							|  |  |  | 		strcpy(menu, "Copy Vert Positions from Shape %t|"); | 
					
						
							|  |  |  | 		for(kb=ky->block.first;kb;kb = kb->next){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			if(a != curshape-1){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				sprintf(menu,"%s %s %cx%d|",menu,kb->name,'%',a); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			a++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | // XXX		nr = pupmenu_col(menu, 20);
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		MEM_freeN(menu); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Object Has No Blendshapes"); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	a = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(kb=ky->block.first;kb;kb = kb->next){ | 
					
						
							|  |  |  | 		if(a == nr){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			for(ev = em->verts.first;ev;ev = ev->next){ | 
					
						
							|  |  |  | 				totverts++; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			if(me->totvert != totverts){ | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 				BKE_report(op->reports, RPT_ERROR, "Shape Has had Verts Added/Removed, please cycle editmode before copying"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				return; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			shape_copy_from_lerp(em, thisBlock,kb); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		a++; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /************************ Merge Operator *************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* Collection Routines|Currently used by the improved merge code*/ | 
					
						
							|  |  |  | /* buildEdge_collection() creates a list of lists*/ | 
					
						
							|  |  |  | /* these lists are filled with edges that are topologically connected.*/ | 
					
						
							|  |  |  | /* This whole tool needs to be redone, its rather poorly implemented...*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct Collection{ | 
					
						
							|  |  |  | 	struct Collection *next, *prev; | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 	ListBase collectionbase; | 
					
						
							|  |  |  | } Collection; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct CollectedEdge{ | 
					
						
							|  |  |  | 	struct CollectedEdge *next, *prev; | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | } CollectedEdge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MERGELIMIT 0.000001
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void build_edgecollection(EditMesh *em, ListBase *allcollections) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	Collection *edgecollection, *newcollection; | 
					
						
							|  |  |  | 	CollectedEdge *newedge; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	int currtag = 1; | 
					
						
							|  |  |  | 	short ebalanced = 0; | 
					
						
							|  |  |  | 	short collectionfound = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (eed=em->edges.first; eed; eed = eed->next){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		eed->tmp.l = 0; | 
					
						
							|  |  |  | 		eed->v1->tmp.l = 0; | 
					
						
							|  |  |  | 		eed->v2->tmp.l = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*1st pass*/ | 
					
						
							|  |  |  | 	for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 			if(eed->f&SELECT){ | 
					
						
							|  |  |  | 				eed->v1->tmp.l = currtag; | 
					
						
							|  |  |  | 				eed->v2->tmp.l = currtag; | 
					
						
							|  |  |  | 				currtag +=1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*2nd pass - Brute force. Loop through selected faces until there are no 'unbalanced' edges left (those with both vertices 'tmp.l' tag matching */ | 
					
						
							|  |  |  | 	while(ebalanced == 0){ | 
					
						
							|  |  |  | 		ebalanced = 1; | 
					
						
							|  |  |  | 		for(eed=em->edges.first; eed; eed = eed->next){ | 
					
						
							|  |  |  | 			if(eed->f&SELECT){ | 
					
						
							|  |  |  | 				if(eed->v1->tmp.l != eed->v2->tmp.l) /*unbalanced*/{ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					if(eed->v1->tmp.l > eed->v2->tmp.l && eed->v2->tmp.l !=0) eed->v1->tmp.l = eed->v2->tmp.l; | 
					
						
							|  |  |  | 					else if(eed->v1 != 0) eed->v2->tmp.l = eed->v1->tmp.l; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					ebalanced = 0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*3rd pass, set all the edge flags (unnessecary?)*/ | 
					
						
							|  |  |  | 	for(eed=em->edges.first; eed; eed = eed->next){ | 
					
						
							|  |  |  | 		if(eed->f&SELECT) eed->tmp.l = eed->v1->tmp.l; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 		if(eed->f&SELECT){ | 
					
						
							|  |  |  | 			if(allcollections->first){ | 
					
						
							|  |  |  | 				for(edgecollection = allcollections->first; edgecollection; edgecollection=edgecollection->next){ | 
					
						
							|  |  |  | 					if(edgecollection->index == eed->tmp.l){ | 
					
						
							|  |  |  | 						newedge = MEM_mallocN(sizeof(CollectedEdge), "collected edge"); | 
					
						
							|  |  |  | 						newedge->eed = eed; | 
					
						
							|  |  |  | 						BLI_addtail(&(edgecollection->collectionbase), newedge); | 
					
						
							|  |  |  | 						collectionfound = 1; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else collectionfound = 0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(allcollections->first == NULL || collectionfound == 0){ | 
					
						
							|  |  |  | 				newcollection = MEM_mallocN(sizeof(Collection), "element collection"); | 
					
						
							|  |  |  | 				newcollection->index = eed->tmp.l; | 
					
						
							|  |  |  | 				newcollection->collectionbase.first = 0; | 
					
						
							|  |  |  | 				newcollection->collectionbase.last = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				newedge = MEM_mallocN(sizeof(CollectedEdge), "collected edge"); | 
					
						
							|  |  |  | 				newedge->eed = eed; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				BLI_addtail(&(newcollection->collectionbase), newedge); | 
					
						
							|  |  |  | 				BLI_addtail(allcollections, newcollection); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void freecollections(ListBase *allcollections) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct Collection *curcollection; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(curcollection = allcollections->first; curcollection; curcollection = curcollection->next) | 
					
						
							|  |  |  | 		BLI_freelistN(&(curcollection->collectionbase)); | 
					
						
							|  |  |  | 	BLI_freelistN(allcollections); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | /*Begin UV Edge Collapse Code
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	Like Edge subdivide, Edge Collapse should handle UV's intelligently, but since UV's are a per-face attribute, normal edge collapse will fail | 
					
						
							|  |  |  | 	in areas such as the boundries of 'UV islands'. So for each edge collection we need to build a set of 'welded' UV vertices and edges for it. | 
					
						
							|  |  |  | 	The welded UV edges can then be sorted and collapsed. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | typedef struct wUV{ | 
					
						
							|  |  |  | 	struct wUV *next, *prev; | 
					
						
							|  |  |  | 	ListBase nodes; | 
					
						
							|  |  |  | 	float u, v; /*cached copy of UV coordinates pointed to by nodes*/ | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	int f; | 
					
						
							|  |  |  | } wUV; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct wUVNode{ | 
					
						
							|  |  |  | 	struct wUVNode *next, *prev; | 
					
						
							|  |  |  | 	float *u; /*pointer to original tface data*/ | 
					
						
							|  |  |  | 	float *v; /*pointer to original tface data*/ | 
					
						
							|  |  |  | } wUVNode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct wUVEdge{ | 
					
						
							|  |  |  | 	struct wUVEdge *next, *prev; | 
					
						
							|  |  |  | 	float v1uv[2], v2uv[2]; /*nasty.*/ | 
					
						
							|  |  |  | 	struct wUV *v1, *v2; /*oriented same as editedge*/ | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	int f; | 
					
						
							|  |  |  | } wUVEdge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct wUVEdgeCollect{ /*used for grouping*/ | 
					
						
							|  |  |  | 	struct wUVEdgeCollect *next, *prev; | 
					
						
							|  |  |  | 	wUVEdge *uved; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	int id; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } wUVEdgeCollect; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void append_weldedUV(EditMesh *em, EditFace *efa, EditVert *eve, int tfindex, ListBase *uvverts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	wUV *curwvert, *newwvert; | 
					
						
							|  |  |  | 	wUVNode *newnode; | 
					
						
							|  |  |  | 	int found; | 
					
						
							|  |  |  | 	MTFace *tf = CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	found = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(curwvert=uvverts->first; curwvert; curwvert=curwvert->next){ | 
					
						
							|  |  |  | 		if(curwvert->eve == eve && curwvert->u == tf->uv[tfindex][0] && curwvert->v == tf->uv[tfindex][1]){ | 
					
						
							|  |  |  | 			newnode = MEM_callocN(sizeof(wUVNode), "Welded UV Vert Node"); | 
					
						
							|  |  |  | 			newnode->u = &(tf->uv[tfindex][0]); | 
					
						
							|  |  |  | 			newnode->v = &(tf->uv[tfindex][1]); | 
					
						
							|  |  |  | 			BLI_addtail(&(curwvert->nodes), newnode); | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!found){ | 
					
						
							|  |  |  | 		newnode = MEM_callocN(sizeof(wUVNode), "Welded UV Vert Node"); | 
					
						
							|  |  |  | 		newnode->u = &(tf->uv[tfindex][0]); | 
					
						
							|  |  |  | 		newnode->v = &(tf->uv[tfindex][1]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		newwvert = MEM_callocN(sizeof(wUV), "Welded UV Vert"); | 
					
						
							|  |  |  | 		newwvert->u = *(newnode->u); | 
					
						
							|  |  |  | 		newwvert->v = *(newnode->v); | 
					
						
							|  |  |  | 		newwvert->eve = eve; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		BLI_addtail(&(newwvert->nodes), newnode); | 
					
						
							|  |  |  | 		BLI_addtail(uvverts, newwvert); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void build_weldedUVs(EditMesh *em, ListBase *uvverts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 		if(efa->v1->f1) append_weldedUV(em, efa, efa->v1, 0, uvverts); | 
					
						
							|  |  |  | 		if(efa->v2->f1) append_weldedUV(em, efa, efa->v2, 1, uvverts); | 
					
						
							|  |  |  | 		if(efa->v3->f1) append_weldedUV(em, efa, efa->v3, 2, uvverts); | 
					
						
							|  |  |  | 		if(efa->v4 && efa->v4->f1) append_weldedUV(em, efa, efa->v4, 3, uvverts); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void append_weldedUVEdge(EditMesh *em, EditFace *efa, EditEdge *eed, ListBase *uvedges) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	wUVEdge *curwedge, *newwedge; | 
					
						
							|  |  |  | 	int v1tfindex, v2tfindex, found; | 
					
						
							|  |  |  | 	MTFace *tf = CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	found = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(eed->v1 == efa->v1) v1tfindex = 0; | 
					
						
							|  |  |  | 	else if(eed->v1 == efa->v2) v1tfindex = 1; | 
					
						
							|  |  |  | 	else if(eed->v1 == efa->v3) v1tfindex = 2; | 
					
						
							|  |  |  | 	else /* if(eed->v1 == efa->v4) */ v1tfindex = 3; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(eed->v2 == efa->v1) v2tfindex = 0; | 
					
						
							|  |  |  | 	else if(eed->v2 == efa->v2) v2tfindex = 1; | 
					
						
							|  |  |  | 	else if(eed->v2 == efa->v3) v2tfindex = 2; | 
					
						
							|  |  |  | 	else /* if(eed->v2 == efa->v4) */ v2tfindex = 3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(curwedge=uvedges->first; curwedge; curwedge=curwedge->next){ | 
					
						
							|  |  |  | 			if(curwedge->eed == eed && curwedge->v1uv[0] == tf->uv[v1tfindex][0] && curwedge->v1uv[1] == tf->uv[v1tfindex][1] && curwedge->v2uv[0] == tf->uv[v2tfindex][0] && curwedge->v2uv[1] == tf->uv[v2tfindex][1]){ | 
					
						
							|  |  |  | 				found = 1; | 
					
						
							|  |  |  | 				break; //do nothing, we don't need another welded uv edge
 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!found){ | 
					
						
							|  |  |  | 		newwedge = MEM_callocN(sizeof(wUVEdge), "Welded UV Edge"); | 
					
						
							|  |  |  | 		newwedge->v1uv[0] = tf->uv[v1tfindex][0]; | 
					
						
							|  |  |  | 		newwedge->v1uv[1] = tf->uv[v1tfindex][1]; | 
					
						
							|  |  |  | 		newwedge->v2uv[0] = tf->uv[v2tfindex][0]; | 
					
						
							|  |  |  | 		newwedge->v2uv[1] = tf->uv[v2tfindex][1]; | 
					
						
							|  |  |  | 		newwedge->eed = eed; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		BLI_addtail(uvedges, newwedge); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void build_weldedUVEdges(EditMesh *em, ListBase *uvedges, ListBase *uvverts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	wUV *curwvert; | 
					
						
							|  |  |  | 	wUVEdge *curwedge; | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 		if(efa->e1->f1) append_weldedUVEdge(em, efa, efa->e1, uvedges); | 
					
						
							|  |  |  | 		if(efa->e2->f1) append_weldedUVEdge(em, efa, efa->e2, uvedges); | 
					
						
							|  |  |  | 		if(efa->e3->f1) append_weldedUVEdge(em, efa, efa->e3, uvedges); | 
					
						
							|  |  |  | 		if(efa->e4 && efa->e4->f1) append_weldedUVEdge(em, efa, efa->e4, uvedges); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	//link vertices: for each uvedge, search uvverts to populate v1 and v2 pointers
 | 
					
						
							|  |  |  | 	for(curwedge=uvedges->first; curwedge; curwedge=curwedge->next){ | 
					
						
							|  |  |  | 		for(curwvert=uvverts->first; curwvert; curwvert=curwvert->next){ | 
					
						
							|  |  |  | 			if(curwedge->eed->v1 == curwvert->eve && curwedge->v1uv[0] == curwvert->u && curwedge->v1uv[1] == curwvert->v){ | 
					
						
							|  |  |  | 				curwedge->v1 = curwvert; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for(curwvert=uvverts->first; curwvert; curwvert=curwvert->next){ | 
					
						
							|  |  |  | 			if(curwedge->eed->v2 == curwvert->eve && curwedge->v2uv[0] == curwvert->u && curwedge->v2uv[1] == curwvert->v){ | 
					
						
							|  |  |  | 				curwedge->v2 = curwvert; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void free_weldedUVs(ListBase *uvverts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	wUV *curwvert; | 
					
						
							|  |  |  | 	for(curwvert = uvverts->first; curwvert; curwvert=curwvert->next) BLI_freelistN(&(curwvert->nodes)); | 
					
						
							|  |  |  | 	BLI_freelistN(uvverts); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void collapse_edgeuvs(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ListBase uvedges, uvverts, allcollections; | 
					
						
							|  |  |  | 	wUVEdge *curwedge; | 
					
						
							|  |  |  | 	wUVNode *curwnode; | 
					
						
							|  |  |  | 	wUVEdgeCollect *collectedwuve, *newcollectedwuve; | 
					
						
							|  |  |  | 	Collection *wuvecollection, *newcollection; | 
					
						
							|  |  |  | 	int curtag, balanced, collectionfound= 0, vcount; | 
					
						
							|  |  |  | 	float avg[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!EM_texFaceCheck(em)) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	uvverts.first = uvverts.last = uvedges.first = uvedges.last = allcollections.first = allcollections.last = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	build_weldedUVs(em, &uvverts); | 
					
						
							|  |  |  | 	build_weldedUVEdges(em, &uvedges, &uvverts); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	curtag = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(curwedge=uvedges.first; curwedge; curwedge=curwedge->next){ | 
					
						
							|  |  |  | 		curwedge->v1->f = curtag; | 
					
						
							|  |  |  | 		curwedge->v2->f = curtag; | 
					
						
							|  |  |  | 		curtag +=1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	balanced = 0; | 
					
						
							|  |  |  | 	while(!balanced){ | 
					
						
							|  |  |  | 		balanced = 1; | 
					
						
							|  |  |  | 		for(curwedge=uvedges.first; curwedge; curwedge=curwedge->next){ | 
					
						
							|  |  |  | 			if(curwedge->v1->f != curwedge->v2->f){ | 
					
						
							|  |  |  | 				if(curwedge->v1->f > curwedge->v2->f) curwedge->v1->f = curwedge->v2->f; | 
					
						
							|  |  |  | 				else curwedge->v2->f = curwedge->v1->f; | 
					
						
							|  |  |  | 				balanced = 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(curwedge=uvedges.first; curwedge; curwedge=curwedge->next) curwedge->f = curwedge->v1->f; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(curwedge=uvedges.first; curwedge; curwedge=curwedge->next){ | 
					
						
							|  |  |  | 		if(allcollections.first){ | 
					
						
							|  |  |  | 			for(wuvecollection = allcollections.first; wuvecollection; wuvecollection=wuvecollection->next){ | 
					
						
							|  |  |  | 				if(wuvecollection->index == curwedge->f){ | 
					
						
							|  |  |  | 					newcollectedwuve = MEM_callocN(sizeof(wUVEdgeCollect), "Collected Welded UV Edge"); | 
					
						
							|  |  |  | 					newcollectedwuve->uved = curwedge; | 
					
						
							|  |  |  | 					BLI_addtail(&(wuvecollection->collectionbase), newcollectedwuve); | 
					
						
							|  |  |  | 					collectionfound = 1; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				else collectionfound = 0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(allcollections.first == NULL || collectionfound == 0){ | 
					
						
							|  |  |  | 			newcollection = MEM_callocN(sizeof(Collection), "element collection"); | 
					
						
							|  |  |  | 			newcollection->index = curwedge->f; | 
					
						
							|  |  |  | 			newcollection->collectionbase.first = 0; | 
					
						
							|  |  |  | 			newcollection->collectionbase.last = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			newcollectedwuve = MEM_callocN(sizeof(wUVEdgeCollect), "Collected Welded UV Edge"); | 
					
						
							|  |  |  | 			newcollectedwuve->uved = curwedge; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			BLI_addtail(&(newcollection->collectionbase), newcollectedwuve); | 
					
						
							|  |  |  | 			BLI_addtail(&allcollections, newcollection); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(wuvecollection=allcollections.first; wuvecollection; wuvecollection=wuvecollection->next){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		vcount = avg[0] = avg[1] = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(collectedwuve= wuvecollection->collectionbase.first; collectedwuve; collectedwuve = collectedwuve->next){ | 
					
						
							|  |  |  | 			avg[0] += collectedwuve->uved->v1uv[0]; | 
					
						
							|  |  |  | 			avg[1] += collectedwuve->uved->v1uv[1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			avg[0] += collectedwuve->uved->v2uv[0]; | 
					
						
							|  |  |  | 			avg[1] += collectedwuve->uved->v2uv[1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			vcount +=2; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		avg[0] /= vcount; avg[1] /= vcount; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(collectedwuve= wuvecollection->collectionbase.first; collectedwuve; collectedwuve = collectedwuve->next){ | 
					
						
							|  |  |  | 			for(curwnode=collectedwuve->uved->v1->nodes.first; curwnode; curwnode=curwnode->next){ | 
					
						
							|  |  |  | 				*(curwnode->u) = avg[0]; | 
					
						
							|  |  |  | 				*(curwnode->v) = avg[1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			for(curwnode=collectedwuve->uved->v2->nodes.first; curwnode; curwnode=curwnode->next){ | 
					
						
							|  |  |  | 				*(curwnode->u) = avg[0]; | 
					
						
							|  |  |  | 				*(curwnode->v) = avg[1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	free_weldedUVs(&uvverts); | 
					
						
							|  |  |  | 	BLI_freelistN(&uvedges); | 
					
						
							|  |  |  | 	freecollections(&allcollections); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*End UV Edge collapse code*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void collapseuvs(EditMesh *em, EditVert *mergevert) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	MTFace *tf; | 
					
						
							|  |  |  | 	int uvcount; | 
					
						
							|  |  |  | 	float uvav[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!EM_texFaceCheck(em)) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	uvcount = 0; | 
					
						
							|  |  |  | 	uvav[0] = 0; | 
					
						
							|  |  |  | 	uvav[1] = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa = em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 		tf = CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(efa->v1->f1 && ELEM(mergevert, NULL, efa->v1)) { | 
					
						
							|  |  |  | 			uvav[0] += tf->uv[0][0]; | 
					
						
							|  |  |  | 			uvav[1] += tf->uv[0][1]; | 
					
						
							|  |  |  | 			uvcount += 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(efa->v2->f1 && ELEM(mergevert, NULL, efa->v2)){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			uvav[0] += tf->uv[1][0]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			uvav[1] += tf->uv[1][1]; | 
					
						
							|  |  |  | 			uvcount += 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(efa->v3->f1 && ELEM(mergevert, NULL, efa->v3)){ | 
					
						
							|  |  |  | 			uvav[0] += tf->uv[2][0]; | 
					
						
							|  |  |  | 			uvav[1] += tf->uv[2][1]; | 
					
						
							|  |  |  | 			uvcount += 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if(efa->v4 && efa->v4->f1 && ELEM(mergevert, NULL, efa->v4)){ | 
					
						
							|  |  |  | 			uvav[0] += tf->uv[3][0]; | 
					
						
							|  |  |  | 			uvav[1] += tf->uv[3][1]; | 
					
						
							|  |  |  | 			uvcount += 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(uvcount > 0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		uvav[0] /= uvcount; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		uvav[1] /= uvcount; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(efa = em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 			tf = CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(efa->v1->f1){ | 
					
						
							|  |  |  | 				tf->uv[0][0] = uvav[0]; | 
					
						
							|  |  |  | 				tf->uv[0][1] = uvav[1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(efa->v2->f1){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 				tf->uv[1][0] = uvav[0]; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tf->uv[1][1] = uvav[1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(efa->v3->f1){ | 
					
						
							|  |  |  | 				tf->uv[2][0] = uvav[0]; | 
					
						
							|  |  |  | 				tf->uv[2][1] = uvav[1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if(efa->v4 && efa->v4->f1){ | 
					
						
							|  |  |  | 				tf->uv[3][0] = uvav[0]; | 
					
						
							|  |  |  | 				tf->uv[3][1] = uvav[1]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int collapseEdges(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	ListBase allcollections; | 
					
						
							|  |  |  | 	CollectedEdge *curredge; | 
					
						
							|  |  |  | 	Collection *edgecollection; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	int totedges, groupcount, mergecount,vcount; | 
					
						
							|  |  |  | 	float avgcount[3]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	allcollections.first = 0; | 
					
						
							|  |  |  | 	allcollections.last = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	mergecount = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	build_edgecollection(em, &allcollections); | 
					
						
							|  |  |  | 	groupcount = BLI_countlist(&allcollections); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(edgecollection = allcollections.first; edgecollection; edgecollection = edgecollection->next){ | 
					
						
							|  |  |  | 		totedges = BLI_countlist(&(edgecollection->collectionbase)); | 
					
						
							|  |  |  | 		mergecount += totedges; | 
					
						
							|  |  |  | 		avgcount[0] = 0; avgcount[1] = 0; avgcount[2] = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		vcount = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(curredge = edgecollection->collectionbase.first; curredge; curredge = curredge->next){ | 
					
						
							|  |  |  | 			avgcount[0] += ((EditEdge*)curredge->eed)->v1->co[0]; | 
					
						
							|  |  |  | 			avgcount[1] += ((EditEdge*)curredge->eed)->v1->co[1]; | 
					
						
							|  |  |  | 			avgcount[2] += ((EditEdge*)curredge->eed)->v1->co[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			avgcount[0] += ((EditEdge*)curredge->eed)->v2->co[0]; | 
					
						
							|  |  |  | 			avgcount[1] += ((EditEdge*)curredge->eed)->v2->co[1]; | 
					
						
							|  |  |  | 			avgcount[2] += ((EditEdge*)curredge->eed)->v2->co[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			vcount +=2; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		avgcount[0] /= vcount; avgcount[1] /=vcount; avgcount[2] /= vcount; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(curredge = edgecollection->collectionbase.first; curredge; curredge = curredge->next){ | 
					
						
							|  |  |  | 			VECCOPY(((EditEdge*)curredge->eed)->v1->co,avgcount); | 
					
						
							|  |  |  | 			VECCOPY(((EditEdge*)curredge->eed)->v2->co,avgcount); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		if (EM_texFaceCheck(em)) { | 
					
						
							|  |  |  | 			/*uv collapse*/ | 
					
						
							|  |  |  | 			for(eve=em->verts.first; eve; eve=eve->next) eve->f1 = 0; | 
					
						
							|  |  |  | 			for(eed=em->edges.first; eed; eed=eed->next) eed->f1 = 0; | 
					
						
							|  |  |  | 			for(curredge = edgecollection->collectionbase.first; curredge; curredge = curredge->next){ | 
					
						
							|  |  |  | 				curredge->eed->v1->f1 = 1; | 
					
						
							|  |  |  | 				curredge->eed->v2->f1 = 1; | 
					
						
							|  |  |  | 				curredge->eed->f1 = 1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			collapse_edgeuvs(em); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	freecollections(&allcollections); | 
					
						
							|  |  |  | 	removedoublesflag(em, 1, 0, MERGELIMIT); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return mergecount; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int merge_firstlast(EditMesh *em, int first, int uvmerge) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve,*mergevert; | 
					
						
							|  |  |  | 	EditSelection *ese; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/* do sanity check in mergemenu in edit.c ?*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	if(first == 0){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		ese = em->selected.last; | 
					
						
							|  |  |  | 		mergevert= (EditVert*)ese->data; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	else{ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		ese = em->selected.first; | 
					
						
							|  |  |  | 		mergevert = (EditVert*)ese->data; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(mergevert->f&SELECT){ | 
					
						
							|  |  |  | 		for (eve=em->verts.first; eve; eve=eve->next){ | 
					
						
							|  |  |  | 			if (eve->f&SELECT) | 
					
						
							|  |  |  | 			VECCOPY(eve->co,mergevert->co); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(uvmerge && CustomData_has_layer(&em->fdata, CD_MTFACE)){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		for(eve=em->verts.first; eve; eve=eve->next) eve->f1 = 0; | 
					
						
							|  |  |  | 		for(eve=em->verts.first; eve; eve=eve->next){ | 
					
						
							|  |  |  | 			if(eve->f&SELECT) eve->f1 = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		collapseuvs(em, mergevert); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	return removedoublesflag(em, 1, 0, MERGELIMIT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-14 05:00:03 +00:00
										 |  |  | void em_snap_to_center(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	float cent[3] = {0.0f, 0.0f, 0.0f}; | 
					
						
							|  |  |  | 	int i=0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (eve=em->verts.first; eve; eve=eve->next) { | 
					
						
							|  |  |  | 		if (eve->f & SELECT) { | 
					
						
							|  |  |  | 			VecAddf(cent, cent, eve->co); | 
					
						
							|  |  |  | 			i++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!i) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	VecMulf(cent, 1.0f / (float)i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (eve=em->verts.first; eve; eve=eve->next) { | 
					
						
							|  |  |  | 		if (eve->f & SELECT) { | 
					
						
							|  |  |  | 			VECCOPY(eve->co, cent); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void em_snap_to_cursor(EditMesh *em, bContext *C) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Scene *scene = CTX_data_scene(C); | 
					
						
							|  |  |  | 	Object *ob= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	View3D *v3d = CTX_wm_view3d(C); | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							|  |  |  | 	float co[3], *vco, invmat[4][4]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	Mat4Invert(invmat, ob->obmat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vco = give_cursor(scene, v3d); | 
					
						
							|  |  |  | 	VECCOPY(co, vco); | 
					
						
							|  |  |  | 	Mat4MulVecfl(invmat, co); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (eve=em->verts.first; eve; eve=eve->next) { | 
					
						
							|  |  |  | 		if (eve->f & SELECT) { | 
					
						
							|  |  |  | 			VECCOPY(eve->co, co); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int merge_target(bContext *C, EditMesh *em, int target, int uvmerge) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	// XXX not working
 | 
					
						
							| 
									
										
										
										
											2009-09-14 05:00:03 +00:00
										 |  |  | 	if(target) em_snap_to_cursor(em, C); | 
					
						
							|  |  |  | 	else em_snap_to_center(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(uvmerge && CustomData_has_layer(&em->fdata, CD_MTFACE)){ | 
					
						
							|  |  |  | 		for(eve=em->verts.first; eve; eve=eve->next) eve->f1 = 0; | 
					
						
							|  |  |  | 		for(eve=em->verts.first; eve; eve=eve->next){ | 
					
						
							|  |  |  | 				if(eve->f&SELECT) eve->f1 = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		collapseuvs(em, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return removedoublesflag(em, 1, 0, MERGELIMIT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #undef MERGELIMIT
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | static int merge_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							|  |  |  | 	int count= 0, uvs= RNA_boolean_get(op->ptr, "uvs"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch(RNA_enum_get(op->ptr, "type")) { | 
					
						
							|  |  |  | 		case 3: | 
					
						
							| 
									
										
										
										
											2009-09-14 05:00:03 +00:00
										 |  |  | 			count = merge_target(C, em, 0, uvs); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case 4: | 
					
						
							| 
									
										
										
										
											2009-09-14 05:00:03 +00:00
										 |  |  | 			count = merge_target(C, em, 1, uvs); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case 1: | 
					
						
							|  |  |  | 			count = merge_firstlast(em, 0, uvs); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 6: | 
					
						
							|  |  |  | 			count = merge_firstlast(em, 1, uvs); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2009-09-14 05:00:03 +00:00
										 |  |  | 		case 5: | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			count = collapseEdges(em); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(!count) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_reportf(op->reports, RPT_INFO, "Removed %d vertices.", count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static EnumPropertyItem merge_type_items[]= { | 
					
						
							|  |  |  | 	{6, "FIRST", 0, "At First", ""}, | 
					
						
							|  |  |  | 	{1, "LAST", 0, "At Last", ""}, | 
					
						
							|  |  |  | 	{3, "CENTER", 0, "At Center", ""}, | 
					
						
							|  |  |  | 	{4, "CURSOR", 0, "At Cursor", ""}, | 
					
						
							|  |  |  | 	{5, "COLLAPSE", 0, "Collapse", ""}, | 
					
						
							|  |  |  | 	{0, NULL, 0, NULL, NULL}}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | static EnumPropertyItem *merge_type_itemf(bContext *C, PointerRNA *ptr, int *free) | 
					
						
							| 
									
										
										
										
											2009-07-13 19:33:59 +00:00
										 |  |  | {	 | 
					
						
							| 
									
										
										
										
											2009-07-14 17:35:07 +00:00
										 |  |  | 	Object *obedit; | 
					
						
							| 
									
										
										
										
											2009-07-13 19:33:59 +00:00
										 |  |  | 	EnumPropertyItem *item= NULL; | 
					
						
							|  |  |  | 	int totitem= 0; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-14 17:35:07 +00:00
										 |  |  | 	if(!C) /* needed for docs */ | 
					
						
							|  |  |  | 		return merge_type_items; | 
					
						
							| 
									
										
										
										
											2009-07-13 19:33:59 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 	if(obedit && obedit->type == OB_MESH) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		EditMesh *em= BKE_mesh_get_editmesh(obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 		if(em->selectmode & SCE_SELECT_VERTEX) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			if(em->selected.first && em->selected.last && | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 				((EditSelection*)em->selected.first)->type == EDITVERT && ((EditSelection*)em->selected.last)->type == EDITVERT) { | 
					
						
							| 
									
										
										
										
											2009-07-14 17:35:07 +00:00
										 |  |  | 				RNA_enum_items_add_value(&item, &totitem, merge_type_items, 6); | 
					
						
							|  |  |  | 				RNA_enum_items_add_value(&item, &totitem, merge_type_items, 1); | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			else if(em->selected.first && ((EditSelection*)em->selected.first)->type == EDITVERT) | 
					
						
							| 
									
										
										
										
											2009-07-14 17:35:07 +00:00
										 |  |  | 				RNA_enum_items_add_value(&item, &totitem, merge_type_items, 1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			else if(em->selected.last && ((EditSelection*)em->selected.last)->type == EDITVERT) | 
					
						
							| 
									
										
										
										
											2009-07-14 17:35:07 +00:00
										 |  |  | 				RNA_enum_items_add_value(&item, &totitem, merge_type_items, 6); | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-14 17:35:07 +00:00
										 |  |  | 		RNA_enum_items_add_value(&item, &totitem, merge_type_items, 3); | 
					
						
							|  |  |  | 		RNA_enum_items_add_value(&item, &totitem, merge_type_items, 4); | 
					
						
							|  |  |  | 		RNA_enum_items_add_value(&item, &totitem, merge_type_items, 5); | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 		RNA_enum_item_end(&item, &totitem); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		*free= 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return item; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-13 19:33:59 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MESH_OT_merge(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	PropertyRNA *prop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Merge"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Merge selected vertices."; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ot->idname= "MESH_OT_merge"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= merge_exec; | 
					
						
							|  |  |  | 	ot->invoke= WM_menu_invoke; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* properties */ | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 	prop= RNA_def_enum(ot->srna, "type", merge_type_items, 3, "Type", "Merge method to use."); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	RNA_def_enum_funcs(prop, merge_type_itemf); | 
					
						
							|  |  |  | 	RNA_def_boolean(ot->srna, "uvs", 0, "UVs", "Move UVs according to merge."); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /************************ Vertex Path Operator *************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PathNode { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	int u; | 
					
						
							|  |  |  | 	int visited; | 
					
						
							|  |  |  | 	ListBase edges; | 
					
						
							|  |  |  | } PathNode; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | typedef struct PathEdge { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	struct PathEdge *next, *prev; | 
					
						
							|  |  |  | 	int v; | 
					
						
							|  |  |  | 	float w; | 
					
						
							|  |  |  | } PathEdge; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | #define PATH_SELECT_EDGE_LENGTH 0
 | 
					
						
							|  |  |  | #define PATH_SELECT_TOPOLOGICAL 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int select_vertex_path_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							|  |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditVert *eve, *s, *t; | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	EditSelection *ese; | 
					
						
							|  |  |  | 	PathEdge *newpe, *currpe; | 
					
						
							|  |  |  | 	PathNode *currpn; | 
					
						
							|  |  |  | 	PathNode *Q; | 
					
						
							|  |  |  | 	int v, *previous, pathvert, pnindex; /*pnindex redundant?*/ | 
					
						
							|  |  |  |  	int unbalanced, totnodes; | 
					
						
							|  |  |  | 	short physical; | 
					
						
							|  |  |  | 	float *cost; | 
					
						
							|  |  |  | 	Heap *heap; /*binary heap for sorting pointers to PathNodes based upon a 'cost'*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	s = t = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	ese = ((EditSelection*)em->selected.last); | 
					
						
							|  |  |  | 	if(ese && ese->type == EDITVERT && ese->prev && ese->prev->type == EDITVERT){ | 
					
						
							|  |  |  | 		physical= pupmenu("Distance Method? %t|Edge Length%x1|Topological%x0"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		t = (EditVert*)ese->data; | 
					
						
							|  |  |  | 		s = (EditVert*)ese->prev->data; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		/*need to find out if t is actually reachable by s....*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 		for(eve=em->verts.first; eve; eve=eve->next){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			eve->f1 = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		s->f1 = 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 		unbalanced = 1; | 
					
						
							|  |  |  | 		totnodes = 1; | 
					
						
							|  |  |  | 		while(unbalanced){ | 
					
						
							|  |  |  | 			unbalanced = 0; | 
					
						
							|  |  |  | 			for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 				if(!eed->h){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					if(eed->v1->f1 && !eed->v2->f1){ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 							eed->v2->f1 = 1; | 
					
						
							|  |  |  | 							totnodes++; | 
					
						
							|  |  |  | 							unbalanced = 1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if(eed->v2->f1 && !eed->v1->f1){ | 
					
						
							|  |  |  | 							eed->v1->f1 = 1; | 
					
						
							|  |  |  | 							totnodes++; | 
					
						
							|  |  |  | 							unbalanced = 1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 		if(s->f1 && t->f1){ /* t can be reached by s */ | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			Q = MEM_callocN(sizeof(PathNode)*totnodes, "Path Select Nodes"); | 
					
						
							|  |  |  | 			totnodes = 0; | 
					
						
							|  |  |  | 			for(eve=em->verts.first; eve; eve=eve->next){ | 
					
						
							|  |  |  | 				if(eve->f1){ | 
					
						
							|  |  |  | 					Q[totnodes].u = totnodes; | 
					
						
							|  |  |  | 					Q[totnodes].edges.first = 0; | 
					
						
							|  |  |  | 					Q[totnodes].edges.last = 0; | 
					
						
							|  |  |  | 					Q[totnodes].visited = 0; | 
					
						
							|  |  |  | 					eve->tmp.p = &(Q[totnodes]); | 
					
						
							|  |  |  | 					totnodes++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else eve->tmp.p = NULL; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 				if(!eed->h){ | 
					
						
							|  |  |  | 					if(eed->v1->f1){ | 
					
						
							|  |  |  | 						currpn = ((PathNode*)eed->v1->tmp.p); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						newpe = MEM_mallocN(sizeof(PathEdge), "Path Edge"); | 
					
						
							|  |  |  | 						newpe->v = ((PathNode*)eed->v2->tmp.p)->u; | 
					
						
							|  |  |  | 						if(physical){ | 
					
						
							|  |  |  | 								newpe->w = VecLenf(eed->v1->co, eed->v2->co); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						else newpe->w = 1; | 
					
						
							|  |  |  | 						newpe->next = 0; | 
					
						
							|  |  |  | 						newpe->prev = 0; | 
					
						
							|  |  |  | 						BLI_addtail(&(currpn->edges), newpe); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					if(eed->v2->f1){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 						currpn = ((PathNode*)eed->v2->tmp.p); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 						newpe = MEM_mallocN(sizeof(PathEdge), "Path Edge"); | 
					
						
							|  |  |  | 						newpe->v = ((PathNode*)eed->v1->tmp.p)->u; | 
					
						
							|  |  |  | 						if(physical){ | 
					
						
							|  |  |  | 								newpe->w = VecLenf(eed->v1->co, eed->v2->co); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						else newpe->w = 1; | 
					
						
							|  |  |  | 						newpe->next = 0; | 
					
						
							|  |  |  | 						newpe->prev = 0; | 
					
						
							|  |  |  | 						BLI_addtail(&(currpn->edges), newpe); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			heap = BLI_heap_new(); | 
					
						
							|  |  |  | 			cost = MEM_callocN(sizeof(float)*totnodes, "Path Select Costs"); | 
					
						
							|  |  |  | 			previous = MEM_callocN(sizeof(int)*totnodes, "PathNode indices"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			for(v=0; v < totnodes; v++){ | 
					
						
							|  |  |  | 				cost[v] = 1000000; | 
					
						
							|  |  |  | 				previous[v] = -1; /*array of indices*/ | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			pnindex = ((PathNode*)s->tmp.p)->u; | 
					
						
							|  |  |  | 			cost[pnindex] = 0; | 
					
						
							|  |  |  | 			BLI_heap_insert(heap,  0.0f, SET_INT_IN_POINTER(pnindex)); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			while( !BLI_heap_empty(heap) ){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				pnindex = GET_INT_FROM_POINTER(BLI_heap_popmin(heap)); | 
					
						
							|  |  |  | 				currpn = &(Q[pnindex]); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(currpn == (PathNode*)t->tmp.p) /*target has been reached....*/ | 
					
						
							|  |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				for(currpe=currpn->edges.first; currpe; currpe=currpe->next){ | 
					
						
							|  |  |  | 					if(!Q[currpe->v].visited){ | 
					
						
							|  |  |  | 						if( cost[currpe->v] > (cost[currpn->u ] + currpe->w) ){ | 
					
						
							|  |  |  | 							cost[currpe->v] = cost[currpn->u] + currpe->w; | 
					
						
							|  |  |  | 							previous[currpe->v] = currpn->u; | 
					
						
							|  |  |  | 							Q[currpe->v].visited = 1; | 
					
						
							|  |  |  | 							BLI_heap_insert(heap, cost[currpe->v], SET_INT_IN_POINTER(currpe->v)); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			pathvert = ((PathNode*)t->tmp.p)->u; | 
					
						
							|  |  |  | 			while(pathvert != -1){ | 
					
						
							|  |  |  | 				for(eve=em->verts.first; eve; eve=eve->next){ | 
					
						
							|  |  |  | 					if(eve->f1){ | 
					
						
							|  |  |  | 						if( ((PathNode*)eve->tmp.p)->u == pathvert) eve->f |= SELECT; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				pathvert = previous[pathvert]; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			for(v=0; v < totnodes; v++) BLI_freelistN(&(Q[v].edges)); | 
					
						
							|  |  |  | 			MEM_freeN(Q); | 
					
						
							|  |  |  | 			MEM_freeN(cost); | 
					
						
							|  |  |  | 			MEM_freeN(previous); | 
					
						
							|  |  |  | 			BLI_heap_free(heap, NULL); | 
					
						
							|  |  |  | 			EM_select_flush(em); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Path Selection requires that exactly two vertices be selected"); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_SELECT, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MESH_OT_select_vertex_path(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static const EnumPropertyItem type_items[] = { | 
					
						
							|  |  |  | 		{PATH_SELECT_EDGE_LENGTH, "EDGE_LENGTH", 0, "Edge Length", NULL}, | 
					
						
							|  |  |  | 		{PATH_SELECT_TOPOLOGICAL, "TOPOLOGICAL", 0, "Topological", NULL}, | 
					
						
							|  |  |  | 		{0, NULL, 0, NULL, NULL}}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Select Vertex Path"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Select shortest path between two vertices by distance type."; | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	ot->idname= "MESH_OT_select_vertex_path"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= select_vertex_path_exec; | 
					
						
							|  |  |  | 	ot->invoke= WM_menu_invoke; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* properties */ | 
					
						
							|  |  |  | 	RNA_def_enum(ot->srna, "type", type_items, PATH_SELECT_EDGE_LENGTH, "Type", "Method to compute distance."); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** Region/Loop Operators *************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int region_to_loop(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int selected= 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	for(eed=em->edges.first; eed; eed=eed->next) eed->f1 = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 		if(efa->f&SELECT){ | 
					
						
							|  |  |  | 			efa->e1->f1++; | 
					
						
							|  |  |  | 			efa->e2->f1++; | 
					
						
							|  |  |  | 			efa->e3->f1++; | 
					
						
							|  |  |  | 			if(efa->e4) | 
					
						
							|  |  |  | 				efa->e4->f1++; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			selected= 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	if(!selected) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	EM_clear_flag_all(em, SELECT); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	for(eed=em->edges.first; eed; eed=eed->next){ | 
					
						
							|  |  |  | 		if(eed->f1 == 1) EM_select_edge(eed, 1); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	em->selectmode = SCE_SELECT_EDGE; | 
					
						
							|  |  |  | 	EM_selectmode_set(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_SELECT, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MESH_OT_region_to_loop(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Region to Loop"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Select a region as a loop of connected edges."; | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	ot->idname= "MESH_OT_region_to_loop"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= region_to_loop; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int validate_loop(EditMesh *em, Collection *edgecollection) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditEdge *eed; | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	CollectedEdge *curredge; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*1st test*/ | 
					
						
							|  |  |  | 	for(curredge = (CollectedEdge*)edgecollection->collectionbase.first; curredge; curredge=curredge->next){ | 
					
						
							|  |  |  | 		curredge->eed->v1->f1 = 0; | 
					
						
							|  |  |  | 		curredge->eed->v2->f1 = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for(curredge = (CollectedEdge*)edgecollection->collectionbase.first; curredge; curredge=curredge->next){ | 
					
						
							|  |  |  | 		curredge->eed->v1->f1++; | 
					
						
							|  |  |  | 		curredge->eed->v2->f1++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for(curredge = (CollectedEdge*)edgecollection->collectionbase.first; curredge; curredge=curredge->next){ | 
					
						
							|  |  |  | 		if(curredge->eed->v1->f1 > 2) return(0); else | 
					
						
							|  |  |  | 		if(curredge->eed->v2->f1 > 2) return(0); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*2nd test*/ | 
					
						
							|  |  |  | 	for(eed = em->edges.first; eed; eed=eed->next) eed->f1 = 0; | 
					
						
							|  |  |  | 	for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 		efa->e1->f1++; | 
					
						
							|  |  |  | 		efa->e2->f1++; | 
					
						
							|  |  |  | 		efa->e3->f1++; | 
					
						
							|  |  |  | 		if(efa->e4) efa->e4->f1++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for(curredge = (CollectedEdge*)edgecollection->collectionbase.first; curredge; curredge=curredge->next){ | 
					
						
							|  |  |  | 		if(curredge->eed->f1 > 2) return(0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return(1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int loop_bisect(EditMesh *em, Collection *edgecollection){ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa, *sf1, *sf2; | 
					
						
							|  |  |  | 	EditEdge *eed, *sed; | 
					
						
							|  |  |  | 	CollectedEdge *curredge; | 
					
						
							|  |  |  | 	int totsf1, totsf2, unbalanced,balancededges; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(eed=em->edges.first; eed; eed=eed->next) eed->f1 = eed->f2 = 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next) efa->f1 = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(curredge = (CollectedEdge*)edgecollection->collectionbase.first; curredge; curredge=curredge->next) curredge->eed->f1 = 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	sf1 = sf2 = NULL; | 
					
						
							|  |  |  | 	sed = ((CollectedEdge*)edgecollection->collectionbase.first)->eed; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 		if(sf2) break; | 
					
						
							|  |  |  | 		else if(sf1){ | 
					
						
							|  |  |  | 			if(efa->e1 == sed || efa->e2 == sed || efa->e3 == sed || ( (efa->e4) ? efa->e4 == sed : 0) ) sf2 = efa; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else{ | 
					
						
							|  |  |  | 			if(efa->e1 == sed || efa->e2 == sed || efa->e3 == sed || ( (efa->e4) ? efa->e4 == sed : 0) ) sf1 = efa; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(sf1==NULL || sf2==NULL) | 
					
						
							|  |  |  | 		return(-1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!(sf1->e1->f1)) sf1->e1->f2 = 1; | 
					
						
							|  |  |  | 	if(!(sf1->e2->f1)) sf1->e2->f2 = 1; | 
					
						
							|  |  |  | 	if(!(sf1->e3->f1)) sf1->e3->f2 = 1; | 
					
						
							|  |  |  | 	if(sf1->e4 && !(sf1->e4->f1)) sf1->e4->f2 = 1; | 
					
						
							|  |  |  | 	sf1->f1 = 1; | 
					
						
							|  |  |  | 	totsf1 = 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(!(sf2->e1->f1)) sf2->e1->f2 = 2; | 
					
						
							|  |  |  | 	if(!(sf2->e2->f1)) sf2->e2->f2 = 2; | 
					
						
							|  |  |  | 	if(!(sf2->e3->f1)) sf2->e3->f2 = 2; | 
					
						
							|  |  |  | 	if(sf2->e4 && !(sf2->e4->f1)) sf2->e4->f2 = 2; | 
					
						
							|  |  |  | 	sf2->f1 = 2; | 
					
						
							|  |  |  | 	totsf2 = 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*do sf1*/ | 
					
						
							|  |  |  | 	unbalanced = 1; | 
					
						
							|  |  |  | 	while(unbalanced){ | 
					
						
							|  |  |  | 		unbalanced = 0; | 
					
						
							|  |  |  | 		for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 			balancededges = 0; | 
					
						
							|  |  |  | 			if(efa->f1 == 0){ | 
					
						
							|  |  |  | 				if(efa->e1->f2 == 1 || efa->e2->f2 == 1 || efa->e3->f2 == 1 || ( (efa->e4) ? efa->e4->f2 == 1 : 0) ){ | 
					
						
							|  |  |  | 					balancededges += efa->e1->f2 = (efa->e1->f1) ? 0 : 1; | 
					
						
							|  |  |  | 					balancededges += efa->e2->f2 = (efa->e2->f1) ? 0 : 1; | 
					
						
							|  |  |  | 					balancededges += efa->e3->f2 = (efa->e3->f1) ? 0 : 1; | 
					
						
							|  |  |  | 					if(efa->e4) balancededges += efa->e4->f2 = (efa->e4->f1) ? 0 : 1; | 
					
						
							|  |  |  | 					if(balancededges){ | 
					
						
							|  |  |  | 						unbalanced = 1; | 
					
						
							|  |  |  | 						efa->f1 = 1; | 
					
						
							|  |  |  | 						totsf1++; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	/*do sf2*/ | 
					
						
							|  |  |  | 	unbalanced = 1; | 
					
						
							|  |  |  | 	while(unbalanced){ | 
					
						
							|  |  |  | 		unbalanced = 0; | 
					
						
							|  |  |  | 		for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 			balancededges = 0; | 
					
						
							|  |  |  | 			if(efa->f1 == 0){ | 
					
						
							|  |  |  | 				if(efa->e1->f2 == 2 || efa->e2->f2 == 2 || efa->e3->f2 == 2 || ( (efa->e4) ? efa->e4->f2 == 2 : 0) ){ | 
					
						
							|  |  |  | 					balancededges += efa->e1->f2 = (efa->e1->f1) ? 0 : 2; | 
					
						
							|  |  |  | 					balancededges += efa->e2->f2 = (efa->e2->f1) ? 0 : 2; | 
					
						
							|  |  |  | 					balancededges += efa->e3->f2 = (efa->e3->f1) ? 0 : 2; | 
					
						
							|  |  |  | 					if(efa->e4) balancededges += efa->e4->f2 = (efa->e4->f1) ? 0 : 2; | 
					
						
							|  |  |  | 					if(balancededges){ | 
					
						
							|  |  |  | 						unbalanced = 1; | 
					
						
							|  |  |  | 						efa->f1 = 2; | 
					
						
							|  |  |  | 						totsf2++; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if(totsf1 < totsf2) return(1); | 
					
						
							|  |  |  | 	else return(2); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int loop_to_region(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 	ListBase allcollections={NULL,NULL}; | 
					
						
							|  |  |  | 	Collection *edgecollection; | 
					
						
							|  |  |  | 	int testflag; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	build_edgecollection(em, &allcollections); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(edgecollection = (Collection *)allcollections.first; edgecollection; edgecollection=edgecollection->next){ | 
					
						
							|  |  |  | 		if(validate_loop(em, edgecollection)){ | 
					
						
							|  |  |  | 			testflag = loop_bisect(em, edgecollection); | 
					
						
							|  |  |  | 			for(efa=em->faces.first; efa; efa=efa->next){ | 
					
						
							|  |  |  | 				if(efa->f1 == testflag){ | 
					
						
							|  |  |  | 					if(efa->f&SELECT) EM_select_face(efa, 0); | 
					
						
							|  |  |  | 					else EM_select_face(efa,1); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next){ /*fix this*/ | 
					
						
							|  |  |  | 		if(efa->f&SELECT) EM_select_face(efa,1); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	freecollections(&allcollections); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_SELECT, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | void MESH_OT_loop_to_region(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Loop to Region"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Select a loop of connected edges as a region."; | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	ot->idname= "MESH_OT_loop_to_region"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= loop_to_region; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** UV/Color Operators *************************/ | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // XXX please check if these functions do what you want them to
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | /* texface and vertex color editmode tools for the face menu */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int mesh_rotate_uvs(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	short change = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MTFace *tf; | 
					
						
							|  |  |  | 	float u1, v1; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int dir= RNA_enum_get(op->ptr, "direction"); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!EM_texFaceCheck(em)) { | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Mesh has no uv/image layers."); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 		if (efa->f & SELECT) { | 
					
						
							|  |  |  | 			tf = CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							|  |  |  | 			u1= tf->uv[0][0]; | 
					
						
							|  |  |  | 			v1= tf->uv[0][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			if (dir == DIRECTION_CCW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					tf->uv[0][0]= tf->uv[3][0]; | 
					
						
							|  |  |  | 					tf->uv[0][1]= tf->uv[3][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[3][0]= tf->uv[2][0]; | 
					
						
							|  |  |  | 					tf->uv[3][1]= tf->uv[2][1]; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					tf->uv[0][0]= tf->uv[2][0]; | 
					
						
							|  |  |  | 					tf->uv[0][1]= tf->uv[2][1]; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tf->uv[2][0]= tf->uv[1][0]; | 
					
						
							|  |  |  | 				tf->uv[2][1]= tf->uv[1][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tf->uv[1][0]= u1; | 
					
						
							|  |  |  | 				tf->uv[1][1]= v1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tf->uv[0][0]= tf->uv[1][0]; | 
					
						
							|  |  |  | 				tf->uv[0][1]= tf->uv[1][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tf->uv[1][0]= tf->uv[2][0]; | 
					
						
							|  |  |  | 				tf->uv[1][1]= tf->uv[2][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					tf->uv[2][0]= tf->uv[3][0]; | 
					
						
							|  |  |  | 					tf->uv[2][1]= tf->uv[3][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[3][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[3][1]= v1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					tf->uv[2][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[2][1]= v1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			change = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(!change) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int mesh_mirror_uvs(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	short change = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MTFace *tf; | 
					
						
							|  |  |  | 	float u1, v1; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int axis= RNA_enum_get(op->ptr, "axis"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (!EM_texFaceCheck(em)) { | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Mesh has no uv/image layers."); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 		if (efa->f & SELECT) { | 
					
						
							|  |  |  | 			tf = CustomData_em_get(&em->fdata, efa->data, CD_MTFACE); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			if (axis == AXIS_Y) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				u1= tf->uv[1][0]; | 
					
						
							|  |  |  | 				v1= tf->uv[1][1]; | 
					
						
							|  |  |  | 				if(efa->v4) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[1][0]= tf->uv[2][0]; | 
					
						
							|  |  |  | 					tf->uv[1][1]= tf->uv[2][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[2][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[2][1]= v1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					u1= tf->uv[3][0]; | 
					
						
							|  |  |  | 					v1= tf->uv[3][1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					tf->uv[3][0]= tf->uv[0][0]; | 
					
						
							|  |  |  | 					tf->uv[3][1]= tf->uv[0][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[0][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[0][1]= v1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					tf->uv[1][0]= tf->uv[2][0]; | 
					
						
							|  |  |  | 					tf->uv[1][1]= tf->uv[2][1]; | 
					
						
							|  |  |  | 					tf->uv[2][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[2][1]= v1; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				u1= tf->uv[0][0]; | 
					
						
							|  |  |  | 				v1= tf->uv[0][1]; | 
					
						
							|  |  |  | 				if(efa->v4) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[0][0]= tf->uv[1][0]; | 
					
						
							|  |  |  | 					tf->uv[0][1]= tf->uv[1][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[1][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[1][1]= v1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					u1= tf->uv[3][0]; | 
					
						
							|  |  |  | 					v1= tf->uv[3][1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					tf->uv[3][0]= tf->uv[2][0]; | 
					
						
							|  |  |  | 					tf->uv[3][1]= tf->uv[2][1]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 					tf->uv[2][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[2][1]= v1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					tf->uv[0][0]= tf->uv[1][0]; | 
					
						
							|  |  |  | 					tf->uv[0][1]= tf->uv[1][1]; | 
					
						
							|  |  |  | 					tf->uv[1][0]= u1; | 
					
						
							|  |  |  | 					tf->uv[1][1]= v1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			change = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(!change) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int mesh_rotate_colors(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	short change = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MCol tmpcol, *mcol; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int dir= RNA_enum_get(op->ptr, "direction"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (!EM_vertColorCheck(em)) { | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Mesh has no color layers."); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 		if (efa->f & SELECT) { | 
					
						
							|  |  |  | 			mcol = CustomData_em_get(&em->fdata, efa->data, CD_MCOL); | 
					
						
							|  |  |  | 			tmpcol= mcol[0]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			if (dir == DIRECTION_CCW) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					mcol[0]= mcol[3]; | 
					
						
							|  |  |  | 					mcol[3]= mcol[2]; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					mcol[0]= mcol[2]; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				mcol[2]= mcol[1]; | 
					
						
							|  |  |  | 				mcol[1]= tmpcol; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				mcol[0]= mcol[1]; | 
					
						
							|  |  |  | 				mcol[1]= mcol[2]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					mcol[2]= mcol[3]; | 
					
						
							|  |  |  | 					mcol[3]= tmpcol; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					mcol[2]= tmpcol; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			change = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(!change) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | static int mesh_mirror_colors(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	EditFace *efa; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	short change = 0; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	MCol tmpcol, *mcol; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int axis= RNA_enum_get(op->ptr, "axis"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	if (!EM_vertColorCheck(em)) { | 
					
						
							| 
									
										
										
										
											2009-02-14 21:31:34 +00:00
										 |  |  | 		BKE_report(op->reports, RPT_ERROR, "Mesh has no color layers"); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 		BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 	for(efa=em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 		if (efa->f & SELECT) { | 
					
						
							|  |  |  | 			mcol = CustomData_em_get(&em->fdata, efa->data, CD_MCOL); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 			if (axis == AXIS_Y) { | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				tmpcol= mcol[1]; | 
					
						
							|  |  |  | 				mcol[1]= mcol[2]; | 
					
						
							|  |  |  | 				mcol[2]= tmpcol; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					tmpcol= mcol[0]; | 
					
						
							|  |  |  | 					mcol[0]= mcol[3]; | 
					
						
							|  |  |  | 					mcol[3]= tmpcol; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				tmpcol= mcol[0]; | 
					
						
							|  |  |  | 				mcol[0]= mcol[1]; | 
					
						
							|  |  |  | 				mcol[1]= tmpcol; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | 				if(efa->v4) { | 
					
						
							|  |  |  | 					tmpcol= mcol[2]; | 
					
						
							|  |  |  | 					mcol[2]= mcol[3]; | 
					
						
							|  |  |  | 					mcol[3]= tmpcol; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			change = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(!change) | 
					
						
							|  |  |  | 		return OPERATOR_CANCELLED; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | void MESH_OT_uvs_rotate(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Rotate UVs"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Rotate selected UVs."; | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | 	ot->idname= "MESH_OT_uvs_rotate"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= mesh_rotate_uvs; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* props */ | 
					
						
							|  |  |  | 	RNA_def_enum(ot->srna, "direction", direction_items, DIRECTION_CW, "Direction", "Direction to rotate UVs around."); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | void MESH_OT_uvs_mirror(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Mirror UVs"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Mirror selected UVs."; | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | 	ot->idname= "MESH_OT_uvs_mirror"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= mesh_mirror_uvs; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	/* props */ | 
					
						
							|  |  |  | 	RNA_def_enum(ot->srna, "axis", axis_items, DIRECTION_CW, "Axis", "Axis to mirror UVs around."); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | void MESH_OT_colors_rotate(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Rotate Colors"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Rotate UV/image color layer."; | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | 	ot->idname= "MESH_OT_colors_rotate"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= mesh_rotate_colors; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	/* props */ | 
					
						
							|  |  |  | 	RNA_def_enum(ot->srna, "direction", direction_items, DIRECTION_CW, "Direction", "Direction to rotate edge around."); | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | void MESH_OT_colors_mirror(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Mirror Colors"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Mirror UV/image color layer."; | 
					
						
							| 
									
										
										
										
											2009-04-12 17:54:07 +00:00
										 |  |  | 	ot->idname= "MESH_OT_colors_mirror"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= mesh_mirror_colors; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 23:20:36 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	/* props */ | 
					
						
							|  |  |  | 	RNA_def_enum(ot->srna, "axis", axis_items, DIRECTION_CW, "Axis", "Axis to mirror colors around."); | 
					
						
							| 
									
										
										
										
											2008-12-30 13:16:14 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** Subdivide Operator *************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | static int subdivide_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	ToolSettings *ts= CTX_data_tool_settings(C); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	int cuts= RNA_int_get(op->ptr,"number_cuts"); | 
					
						
							|  |  |  | 	float smooth= 0.292f*RNA_float_get(op->ptr, "smoothness"); | 
					
						
							|  |  |  | 	float fractal= RNA_float_get(op->ptr, "fractal")/100; | 
					
						
							|  |  |  | 	int flag= 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(smooth != 0.0f) | 
					
						
							|  |  |  | 		flag |= B_SMOOTH; | 
					
						
							|  |  |  | 	if(fractal != 0.0f) | 
					
						
							|  |  |  | 		flag |= B_FRACTAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	esubdivideflag(obedit, em, 1, smooth, fractal, ts->editbutflag|flag, cuts, 0); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MESH_OT_subdivide(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Subdivide"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Subdivide selected edges."; | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | 	ot->idname= "MESH_OT_subdivide"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= subdivide_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-16 23:53:11 +00:00
										 |  |  | 	/* properties */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	RNA_def_int(ot->srna, "number_cuts", 1, 1, 10, "Number of Cuts", "", 1, INT_MAX); | 
					
						
							|  |  |  | 	RNA_def_float(ot->srna, "fractal", 0.0, 0.0f, FLT_MAX, "Fractal", "Fractal randomness factor.", 0.0f, 1000.0f); | 
					
						
							|  |  |  | 	RNA_def_float(ot->srna, "smoothness", 0.0f, 0.0f, 1000.0f, "Smoothness", "Smoothness factor.", 0.0f, FLT_MAX); | 
					
						
							| 
									
										
										
										
											2009-01-16 04:48:33 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** Fill Operators *************************/ | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* note; the EM_selectmode_set() calls here illustrate how badly constructed it all is... from before the
 | 
					
						
							|  |  |  | edge/face flags, with very mixed results.... */ | 
					
						
							|  |  |  | static void beauty_fill(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *v1, *v2, *v3, *v4; | 
					
						
							|  |  |  | 	EditEdge *eed, *nexted; | 
					
						
							|  |  |  | 	EditEdge dia1, dia2; | 
					
						
							|  |  |  | 	EditFace *efa, *w; | 
					
						
							|  |  |  | 	// void **efaar, **efaa;
 | 
					
						
							|  |  |  | 	EVPTuple *efaar; | 
					
						
							|  |  |  | 	EVPtr *efaa; | 
					
						
							|  |  |  | 	float len1, len2, len3, len4, len5, len6, opp1, opp2, fac1, fac2; | 
					
						
							|  |  |  | 	int totedge, ok, notbeauty=8, onedone, vindex[4]; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	/* - all selected edges with two faces
 | 
					
						
							|  |  |  | 		* - find the faces: store them in edges (using datablock) | 
					
						
							|  |  |  | 		* - per edge: - test convex | 
					
						
							|  |  |  | 		*			   - test edge: flip? | 
					
						
							|  |  |  | 		*			   - if true: remedge,  addedge, all edges at the edge get new face pointers | 
					
						
							|  |  |  | 		*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	EM_selectmode_set(em);	// makes sure in selectmode 'face' the edges of selected faces are selected too
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	totedge = count_selected_edges(em->edges.first); | 
					
						
							|  |  |  | 	if(totedge==0) return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	/* temp block with face pointers */ | 
					
						
							|  |  |  | 	efaar= (EVPTuple *) MEM_callocN(totedge * sizeof(EVPTuple), "beautyfill"); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	while (notbeauty) { | 
					
						
							|  |  |  | 		notbeauty--; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		ok = collect_quadedges(efaar, em->edges.first, em->faces.first); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		/* there we go */ | 
					
						
							|  |  |  | 		onedone= 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		eed= em->edges.first; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			nexted= eed->next; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			/* f2 is set in collect_quadedges() */ | 
					
						
							|  |  |  | 			if(eed->f2==2 && eed->h==0) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 				efaa = (EVPtr *) eed->tmp.p; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 				/* none of the faces should be treated before, nor be part of fgon */ | 
					
						
							|  |  |  | 				ok= 1; | 
					
						
							|  |  |  | 				efa= efaa[0]; | 
					
						
							|  |  |  | 				if(efa->e1->f1 || efa->e2->f1 || efa->e3->f1) ok= 0; | 
					
						
							|  |  |  | 				if(efa->fgonf) ok= 0; | 
					
						
							|  |  |  | 				efa= efaa[1]; | 
					
						
							|  |  |  | 				if(efa->e1->f1 || efa->e2->f1 || efa->e3->f1) ok= 0; | 
					
						
							|  |  |  | 				if(efa->fgonf) ok= 0; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 				if(ok) { | 
					
						
							|  |  |  | 					/* test convex */ | 
					
						
							|  |  |  | 					givequadverts(efaa[0], efaa[1], &v1, &v2, &v3, &v4, vindex); | 
					
						
							|  |  |  | 					if(v1 && v2 && v3 && v4) { | 
					
						
							|  |  |  | 						if( convex(v1->co, v2->co, v3->co, v4->co) ) { | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							/* test edges */ | 
					
						
							|  |  |  | 							if( (v1) > (v3) ) { | 
					
						
							|  |  |  | 								dia1.v1= v3; | 
					
						
							|  |  |  | 								dia1.v2= v1; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								dia1.v1= v1; | 
					
						
							|  |  |  | 								dia1.v2= v3; | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							if( (v2) > (v4) ) { | 
					
						
							|  |  |  | 								dia2.v1= v4; | 
					
						
							|  |  |  | 								dia2.v2= v2; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								dia2.v1= v2; | 
					
						
							|  |  |  | 								dia2.v2= v4; | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							/* testing rule:
 | 
					
						
							|  |  |  | 							* the area divided by the total edge lengths | 
					
						
							|  |  |  | 							*/ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							len1= VecLenf(v1->co, v2->co); | 
					
						
							|  |  |  | 							len2= VecLenf(v2->co, v3->co); | 
					
						
							|  |  |  | 							len3= VecLenf(v3->co, v4->co); | 
					
						
							|  |  |  | 							len4= VecLenf(v4->co, v1->co); | 
					
						
							|  |  |  | 							len5= VecLenf(v1->co, v3->co); | 
					
						
							|  |  |  | 							len6= VecLenf(v2->co, v4->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							opp1= AreaT3Dfl(v1->co, v2->co, v3->co); | 
					
						
							|  |  |  | 							opp2= AreaT3Dfl(v1->co, v3->co, v4->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							fac1= opp1/(len1+len2+len5) + opp2/(len3+len4+len5); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							opp1= AreaT3Dfl(v2->co, v3->co, v4->co); | 
					
						
							|  |  |  | 							opp2= AreaT3Dfl(v2->co, v4->co, v1->co); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							fac2= opp1/(len2+len3+len6) + opp2/(len4+len1+len6); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							ok= 0; | 
					
						
							|  |  |  | 							if(fac1 > fac2) { | 
					
						
							|  |  |  | 								if(dia2.v1==eed->v1 && dia2.v2==eed->v2) { | 
					
						
							|  |  |  | 									eed->f1= 1; | 
					
						
							|  |  |  | 									efa= efaa[0]; | 
					
						
							|  |  |  | 									efa->f1= 1; | 
					
						
							|  |  |  | 									efa= efaa[1]; | 
					
						
							|  |  |  | 									efa->f1= 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 									w= EM_face_from_faces(em, efaa[0], efaa[1], | 
					
						
							|  |  |  | 														  vindex[0], vindex[1], 4+vindex[2], -1); | 
					
						
							|  |  |  | 									w->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 									w= EM_face_from_faces(em, efaa[0], efaa[1], | 
					
						
							|  |  |  | 														  vindex[0], 4+vindex[2], 4+vindex[3], -1); | 
					
						
							|  |  |  | 									w->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 									onedone= 1; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else if(fac1 < fac2) { | 
					
						
							|  |  |  | 								if(dia1.v1==eed->v1 && dia1.v2==eed->v2) { | 
					
						
							|  |  |  | 									eed->f1= 1; | 
					
						
							|  |  |  | 									efa= efaa[0]; | 
					
						
							|  |  |  | 									efa->f1= 1; | 
					
						
							|  |  |  | 									efa= efaa[1]; | 
					
						
							|  |  |  | 									efa->f1= 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 									w= EM_face_from_faces(em, efaa[0], efaa[1], | 
					
						
							|  |  |  | 														  vindex[1], 4+vindex[2], 4+vindex[3], -1); | 
					
						
							|  |  |  | 									w->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 									w= EM_face_from_faces(em, efaa[0], efaa[1], | 
					
						
							|  |  |  | 														  vindex[0], 4+vindex[1], 4+vindex[3], -1); | 
					
						
							|  |  |  | 									w->f |= SELECT; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 									onedone= 1; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			eed= nexted; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		free_tagged_edges_faces(em, em->edges.first, em->faces.first); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		if(onedone==0) break; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		EM_selectmode_set(em);	// new edges/faces were added
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	MEM_freeN(efaar); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	EM_select_flush(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Got this from scanfill.c. You will need to juggle around the
 | 
					
						
							|  |  |  | * callbacks for the scanfill.c code a bit for this to work. */ | 
					
						
							|  |  |  | static void fill_mesh(EditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditVert *eve,*v1; | 
					
						
							|  |  |  | 	EditEdge *eed,*e1,*nexted; | 
					
						
							|  |  |  | 	EditFace *efa,*nextvl, *efan; | 
					
						
							|  |  |  | 	short ok; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	if(em==NULL) return; | 
					
						
							|  |  |  | 	waitcursor(1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	/* copy all selected vertices */ | 
					
						
							|  |  |  | 	eve= em->verts.first; | 
					
						
							|  |  |  | 	while(eve) { | 
					
						
							|  |  |  | 		if(eve->f & SELECT) { | 
					
						
							|  |  |  | 			v1= BLI_addfillvert(eve->co); | 
					
						
							|  |  |  | 			eve->tmp.v= v1; | 
					
						
							|  |  |  | 			v1->tmp.v= eve; | 
					
						
							|  |  |  | 			v1->xs= 0;	// used for counting edges
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eve= eve->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* copy all selected edges */ | 
					
						
							|  |  |  | 	eed= em->edges.first; | 
					
						
							|  |  |  | 	while(eed) { | 
					
						
							|  |  |  | 		if( (eed->v1->f & SELECT) && (eed->v2->f & SELECT) ) { | 
					
						
							|  |  |  | 			e1= BLI_addfilledge(eed->v1->tmp.v, eed->v2->tmp.v); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			e1->v1->xs++; | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 			e1->v2->xs++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		eed= eed->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* from all selected faces: remove vertices and edges to prevent doubles */ | 
					
						
							|  |  |  | 	/* all edges add values, faces subtract,
 | 
					
						
							|  |  |  | 		then remove edges with vertices ->xs<2 */ | 
					
						
							|  |  |  | 	efa= em->faces.first; | 
					
						
							|  |  |  | 	ok= 0; | 
					
						
							|  |  |  | 	while(efa) { | 
					
						
							|  |  |  | 		nextvl= efa->next; | 
					
						
							|  |  |  | 		if( faceselectedAND(efa, 1) ) { | 
					
						
							|  |  |  | 			efa->v1->tmp.v->xs--; | 
					
						
							|  |  |  | 			efa->v2->tmp.v->xs--; | 
					
						
							|  |  |  | 			efa->v3->tmp.v->xs--; | 
					
						
							|  |  |  | 			if(efa->v4) efa->v4->tmp.v->xs--; | 
					
						
							|  |  |  | 			ok= 1; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		efa= nextvl; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if(ok) {	/* there are faces selected */ | 
					
						
							|  |  |  | 		eed= filledgebase.first; | 
					
						
							|  |  |  | 		while(eed) { | 
					
						
							|  |  |  | 			nexted= eed->next; | 
					
						
							|  |  |  | 			if(eed->v1->xs<2 || eed->v2->xs<2) { | 
					
						
							|  |  |  | 				BLI_remlink(&filledgebase,eed); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			eed= nexted; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(BLI_edgefill(0, em->mat_nr)) { | 
					
						
							|  |  |  | 		efa= fillfacebase.first; | 
					
						
							|  |  |  | 		while(efa) { | 
					
						
							|  |  |  | 			/* normals default pointing up */ | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 			efan= addfacelist(em, efa->v3->tmp.v, efa->v2->tmp.v, | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 							  efa->v1->tmp.v, 0, NULL, NULL); | 
					
						
							|  |  |  | 			if(efan) EM_select_face(efan, 1); | 
					
						
							|  |  |  | 			efa= efa->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_end_edgefill(); | 
					
						
							|  |  |  | 	beauty_fill(em); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	WM_cursor_wait(0); | 
					
						
							|  |  |  | 	EM_select_flush(em); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | static int fill_mesh_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	fill_mesh(em); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | void MESH_OT_fill(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ot->name= "Fill"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Create a segment, edge or face."; | 
					
						
							| 
									
										
										
										
											2009-02-01 12:40:27 +00:00
										 |  |  | 	ot->idname= "MESH_OT_fill"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= fill_mesh_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int beauty_fill_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 19:03:53 +00:00
										 |  |  | 	beauty_fill(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MESH_OT_beauty_fill(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Beauty Fill"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Arrange geometry on a selected surface to avoid skinny faces."; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->idname= "MESH_OT_beauty_fill"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= beauty_fill_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** Quad/Tri Operators *************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | static int quads_convert_to_tris_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	convert_to_triface(em,0); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | void MESH_OT_quads_convert_to_tris(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Quads to Tris"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Convert selected quads to triangles."; | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | 	ot->idname= "MESH_OT_quads_convert_to_tris"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | 	ot->exec= quads_convert_to_tris_exec; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | static int tris_convert_to_quads_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	join_triangles(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | void MESH_OT_tris_convert_to_quads(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Tris to Quads"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Convert selected triangles to quads."; | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | 	ot->idname= "MESH_OT_tris_convert_to_quads"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-04-15 15:40:31 +00:00
										 |  |  | 	ot->exec= tris_convert_to_quads_exec; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int edge_flip_exec(bContext *C, wmOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	edge_flip(em); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MESH_OT_edge_flip(wmOperatorType *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							|  |  |  | 	ot->name= "Edge Flip"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Flip selected edge or adjoining faces."; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->idname= "MESH_OT_edge_flip"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							|  |  |  | 	ot->exec= edge_flip_exec; | 
					
						
							|  |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | /********************** Smooth/Solid Operators *************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void mesh_set_smooth_faces(EditMesh *em, short smooth) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	EditFace *efa; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(em==NULL) return; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	for(efa= em->faces.first; efa; efa=efa->next) { | 
					
						
							|  |  |  | 		if(efa->f & SELECT) { | 
					
						
							|  |  |  | 			if(smooth) efa->flag |= ME_SMOOTH; | 
					
						
							|  |  |  | 			else efa->flag &= ~ME_SMOOTH; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 04:22:18 +00:00
										 |  |  | static int mesh_faces_shade_smooth_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	mesh_set_smooth_faces(em, 1); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 	BKE_mesh_end_editmesh(obedit->data, em); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 04:22:18 +00:00
										 |  |  | void MESH_OT_faces_shade_smooth(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ot->name= "Shade Smooth"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Display faces 'smooth' (using vertex normals)."; | 
					
						
							| 
									
										
										
										
											2009-02-01 04:22:18 +00:00
										 |  |  | 	ot->idname= "MESH_OT_faces_shade_smooth"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-02-01 04:22:18 +00:00
										 |  |  | 	ot->exec= mesh_faces_shade_smooth_exec; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-21 00:36:07 +00:00
										 |  |  | static int mesh_faces_shade_flat_exec(bContext *C, wmOperator *op) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Object *obedit= CTX_data_edit_object(C); | 
					
						
							| 
									
										
										
										
											2009-04-11 08:26:51 +00:00
										 |  |  | 	EditMesh *em= BKE_mesh_get_editmesh((Mesh *)obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	mesh_set_smooth_faces(em, 0); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												2.5
Notifiers
---------
Various fixes for wrong use of notifiers, and some new notifiers
to make things a bit more clear and consistent, with two notable
changes:
* Geometry changes are now done with NC_GEOM, rather than
  NC_OBJECT|ND_GEOM_, so an object does need to be available.
* Space data now use NC_SPACE|ND_SPACE_*, instead of data
  notifiers or even NC_WINDOW in some cases. Note that NC_SPACE
  should only be used for notifying about changes in space data,
  we don't want to go back to allqueue(REDRAW..).
Depsgraph
---------
The dependency graph now has a different flush call:
DAG_object_flush_update(scene, ob, flag)
is replaced by:
DAG_id_flush_update(id, flag)
It still works basically the same, one difference is that it now
also accepts object data (e.g. Mesh), again to avoid requiring an
Object to be available. Other ID types will simply do nothing at
the moment.
Docs
----
I made some guidelines for how/when to do which kinds of updates
and notifiers. I can't specify totally exact how to make these
decisions, but these are basically the guidelines I use. So, new
and updated docs are here:
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/NotifiersUpdates
http://wiki.blender.org/index.php/BlenderDev/Blender2.5/DataNotifiers
											
										 
											2009-09-04 20:51:09 +00:00
										 |  |  | 	DAG_id_flush_update(obedit->data, OB_RECALC_DATA); | 
					
						
							|  |  |  | 	WM_event_add_notifier(C, NC_GEOM|ND_DATA, obedit->data); | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	return OPERATOR_FINISHED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-21 00:36:07 +00:00
										 |  |  | void MESH_OT_faces_shade_flat(wmOperatorType *ot) | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* identifiers */ | 
					
						
							| 
									
										
										
										
											2009-07-08 21:31:28 +00:00
										 |  |  | 	ot->name= "Shade Flat"; | 
					
						
							| 
									
										
										
										
											2009-08-17 04:15:53 +00:00
										 |  |  | 	ot->description= "Display faces 'flat'."; | 
					
						
							| 
									
										
										
										
											2009-07-21 00:36:07 +00:00
										 |  |  | 	ot->idname= "MESH_OT_faces_shade_flat"; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	/* api callbacks */ | 
					
						
							| 
									
										
										
										
											2009-07-21 00:36:07 +00:00
										 |  |  | 	ot->exec= mesh_faces_shade_flat_exec; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | 	ot->poll= ED_operator_editmesh; | 
					
						
							| 
									
										
										
										
											2009-07-12 02:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-31 19:40:40 +00:00
										 |  |  | 	/* flags */ | 
					
						
							|  |  |  | 	ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO; | 
					
						
							| 
									
										
										
										
											2009-01-24 22:21:12 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-07-08 15:34:41 +00:00
										 |  |  | 
 |