| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Joseph Eagar. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  | /** \file blender/bmesh/operators/bmo_dupe.c
 | 
					
						
							|  |  |  |  *  \ingroup bmesh | 
					
						
							| 
									
										
										
										
											2013-03-30 08:54:50 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-06-25 22:58:23 +00:00
										 |  |  |  * Duplicate, Split, Split operators. | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2013-07-28 10:38:25 +00:00
										 |  |  | #include "BLI_alloca.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "bmesh.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  | #include "intern/bmesh_operators_private.h" /* own include */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | /* local flag define */ | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | #define DUPE_INPUT      1 /* input from operator */
 | 
					
						
							|  |  |  | #define DUPE_NEW        2
 | 
					
						
							|  |  |  | #define DUPE_DONE       4
 | 
					
						
							| 
									
										
										
										
											2012-09-20 01:02:39 +00:00
										 |  |  | // #define DUPE_MAPPED     8 // UNUSED
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * COPY VERTEX | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  |  * Copy an existing vertex from one bmesh to another. | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | static BMVert *copy_vertex(BMesh *source_mesh, BMVert *source_vertex, BMesh *target_mesh, GHash *vhash) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMVert *target_vertex = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* Create a new vertex */ | 
					
						
							| 
									
										
										
										
											2012-11-29 16:26:39 +00:00
										 |  |  | 	target_vertex = BM_vert_create(target_mesh, source_vertex->co, NULL, BM_CREATE_SKIP_CD); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* Insert new vertex into the vert hash */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BLI_ghash_insert(vhash, source_vertex, target_vertex); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* Copy attributes */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BM_elem_attrs_copy(source_mesh, target_mesh, source_vertex, target_vertex); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* Set internal op flags */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMO_elem_flag_enable(target_mesh, target_vertex, DUPE_NEW); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return target_vertex; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * COPY EDGE | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copy an existing edge from one bmesh to another. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | static BMEdge *copy_edge(BMOperator *op, | 
					
						
							|  |  |  |                          BMOpSlot *slot_boundarymap_out, | 
					
						
							|  |  |  |                          BMesh *source_mesh, | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |                          BMEdge *source_edge, BMesh *target_mesh, | 
					
						
							|  |  |  |                          GHash *vhash, GHash *ehash) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMEdge *target_edge = NULL; | 
					
						
							|  |  |  | 	BMVert *target_vert1, *target_vert2; | 
					
						
							|  |  |  | 	BMFace *face; | 
					
						
							|  |  |  | 	BMIter fiter; | 
					
						
							|  |  |  | 	int rlen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* see if any of the neighboring faces are
 | 
					
						
							|  |  |  | 	 * not being duplicated.  in that case, | 
					
						
							|  |  |  | 	 * add it to the new/old map. */ | 
					
						
							|  |  |  | 	rlen = 0; | 
					
						
							|  |  |  | 	for (face = BM_iter_new(&fiter, source_mesh, BM_FACES_OF_EDGE, source_edge); | 
					
						
							|  |  |  | 	     face; | 
					
						
							|  |  |  | 	     face = BM_iter_step(&fiter)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(source_mesh, face, DUPE_INPUT)) { | 
					
						
							|  |  |  | 			rlen++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Lookup v1 and v2 */ | 
					
						
							|  |  |  | 	target_vert1 = BLI_ghash_lookup(vhash, source_edge->v1); | 
					
						
							|  |  |  | 	target_vert2 = BLI_ghash_lookup(vhash, source_edge->v2); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* Create a new edge */ | 
					
						
							| 
									
										
										
										
											2012-11-29 16:26:39 +00:00
										 |  |  | 	target_edge = BM_edge_create(target_mesh, target_vert1, target_vert2, NULL, BM_CREATE_SKIP_CD); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* add to new/old edge map if necassary */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	if (rlen < 2) { | 
					
						
							|  |  |  | 		/* not sure what non-manifold cases of greater then three
 | 
					
						
							|  |  |  | 		 * radial should do. */ | 
					
						
							| 
									
										
										
										
											2012-11-26 03:16:29 +00:00
										 |  |  | 		BMO_slot_map_elem_insert(op, slot_boundarymap_out, source_edge, target_edge); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Insert new edge into the edge hash */ | 
					
						
							|  |  |  | 	BLI_ghash_insert(ehash, source_edge, target_edge); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* Copy attributes */ | 
					
						
							|  |  |  | 	BM_elem_attrs_copy(source_mesh, target_mesh, source_edge, target_edge); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* Set internal op flags */ | 
					
						
							|  |  |  | 	BMO_elem_flag_enable(target_mesh, target_edge, DUPE_NEW); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return target_edge; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * COPY FACE | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  |  * Copy an existing face from one bmesh to another. | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | static BMFace *copy_face(BMOperator *op, | 
					
						
							|  |  |  |                          BMOpSlot *slot_facemap_out, | 
					
						
							|  |  |  |                          BMesh *source_mesh, | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |                          BMFace *source_face, BMesh *target_mesh, | 
					
						
							| 
									
										
										
										
											2013-07-19 10:39:32 +00:00
										 |  |  |                          GHash *vhash, GHash *ehash) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* BMVert *target_vert1, *target_vert2; */ /* UNUSED */ | 
					
						
							| 
									
										
										
										
											2013-07-19 10:39:32 +00:00
										 |  |  | 	BMVert **vtar = BLI_array_alloca(vtar, source_face->len); | 
					
						
							|  |  |  | 	BMEdge **edar = BLI_array_alloca(edar, source_face->len); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMLoop *source_loop, *target_loop; | 
					
						
							|  |  |  | 	BMFace *target_face = NULL; | 
					
						
							|  |  |  | 	BMIter iter, iter2; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* lookup the first and second vert */ | 
					
						
							|  |  |  | #if 0 /* UNUSED */
 | 
					
						
							|  |  |  | 	target_vert1 = BLI_ghash_lookup(vhash, BM_iter_new(&iter, source_mesh, BM_VERTS_OF_FACE, source_face)); | 
					
						
							|  |  |  | 	target_vert2 = BLI_ghash_lookup(vhash, BM_iter_step(&iter)); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	BM_iter_new(&iter, source_mesh, BM_VERTS_OF_FACE, source_face); | 
					
						
							|  |  |  | 	BM_iter_step(&iter); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* lookup edge */ | 
					
						
							| 
									
										
										
										
											2013-07-24 18:38:55 +00:00
										 |  |  | 	BM_ITER_ELEM_INDEX (source_loop, &iter, source_face, BM_LOOPS_OF_FACE, i) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		vtar[i] = BLI_ghash_lookup(vhash, source_loop->v); | 
					
						
							|  |  |  | 		edar[i] = BLI_ghash_lookup(ehash, source_loop->e); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* create new face */ | 
					
						
							| 
									
										
										
										
											2013-08-21 11:09:50 +00:00
										 |  |  | 	target_face = BM_face_create(target_mesh, vtar, edar, source_face->len, source_face, BM_CREATE_NOP); | 
					
						
							| 
									
										
										
										
											2012-11-26 03:16:29 +00:00
										 |  |  | 	BMO_slot_map_elem_insert(op, slot_facemap_out, source_face, target_face); | 
					
						
							|  |  |  | 	BMO_slot_map_elem_insert(op, slot_facemap_out, target_face, source_face); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* mark the face for output */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMO_elem_flag_enable(target_mesh, target_face, DUPE_NEW); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* copy per-loop custom data */ | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 	BM_ITER_ELEM (source_loop, &iter, source_face, BM_LOOPS_OF_FACE) { | 
					
						
							|  |  |  | 		BM_ITER_ELEM (target_loop, &iter2, target_face, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			if (BLI_ghash_lookup(vhash, source_loop->v) == target_loop->v) { | 
					
						
							|  |  |  | 				BM_elem_attrs_copy(source_mesh, target_mesh, source_loop, target_loop); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return target_face; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * COPY MESH | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Internal Copy function. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | static void bmo_mesh_copy(BMOperator *op, BMesh *bm_src, BMesh *bm_dst) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BMVert *v = NULL, *v2; | 
					
						
							|  |  |  | 	BMEdge *e = NULL; | 
					
						
							|  |  |  | 	BMFace *f = NULL; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-31 12:29:17 +00:00
										 |  |  | 	BMIter viter, eiter, fiter; | 
					
						
							|  |  |  | 	GHash *vhash, *ehash; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMOpSlot *slot_boundary_map_out = BMO_slot_get(op->slots_out, "boundary_map.out"); | 
					
						
							|  |  |  | 	BMOpSlot *slot_face_map_out     = BMO_slot_get(op->slots_out, "face_map.out"); | 
					
						
							|  |  |  | 	BMOpSlot *slot_isovert_map_out  = BMO_slot_get(op->slots_out, "isovert_map.out"); | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* initialize pointer hashes */ | 
					
						
							| 
									
										
										
										
											2012-05-16 00:51:36 +00:00
										 |  |  | 	vhash = BLI_ghash_ptr_new("bmesh dupeops v"); | 
					
						
							|  |  |  | 	ehash = BLI_ghash_ptr_new("bmesh dupeops e"); | 
					
						
							| 
									
										
										
										
											2012-03-27 12:34:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* duplicate flagged vertices */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 	BM_ITER_MESH (v, &viter, bm_src, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(bm_src, v, DUPE_INPUT) && | 
					
						
							|  |  |  | 		    !BMO_elem_flag_test(bm_src, v, DUPE_DONE)) | 
					
						
							| 
									
										
										
										
											2012-04-01 15:25:07 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMIter iter; | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 			bool isolated = true; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 			v2 = copy_vertex(bm_src, v, bm_dst, vhash); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (f, &iter, v, BM_FACES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 				if (BMO_elem_flag_test(bm_src, f, DUPE_INPUT)) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 					isolated = false; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-27 12:34:00 +00:00
										 |  |  | 			if (isolated) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (e, &iter, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 					if (BMO_elem_flag_test(bm_src, e, DUPE_INPUT)) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 						isolated = false; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-27 12:34:00 +00:00
										 |  |  | 			if (isolated) { | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 				BMO_slot_map_elem_insert(op, slot_isovert_map_out, v, v2); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 			BMO_elem_flag_enable(bm_src, v, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* now we dupe all the edges */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 	BM_ITER_MESH (e, &eiter, bm_src, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(bm_src, e, DUPE_INPUT) && | 
					
						
							|  |  |  | 		    !BMO_elem_flag_test(bm_src, e, DUPE_DONE)) | 
					
						
							| 
									
										
										
										
											2012-04-01 15:25:07 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 			/* make sure that verts are copied */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 			if (!BMO_elem_flag_test(bm_src, e->v1, DUPE_DONE)) { | 
					
						
							|  |  |  | 				copy_vertex(bm_src, e->v1, bm_dst, vhash); | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm_src, e->v1, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 			if (!BMO_elem_flag_test(bm_src, e->v2, DUPE_DONE)) { | 
					
						
							|  |  |  | 				copy_vertex(bm_src, e->v2, bm_dst, vhash); | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm_src, e->v2, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 			/* now copy the actual edge */ | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 			copy_edge(op, slot_boundary_map_out, bm_src, e, bm_dst, vhash, ehash); | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 			BMO_elem_flag_enable(bm_src, e, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* first we dupe all flagged faces and their elements from source */ | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 	BM_ITER_MESH (f, &fiter, bm_src, BM_FACES_OF_MESH) { | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(bm_src, f, DUPE_INPUT)) { | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 			/* vertex pass */ | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (v, &viter, f, BM_VERTS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 				if (!BMO_elem_flag_test(bm_src, v, DUPE_DONE)) { | 
					
						
							|  |  |  | 					copy_vertex(bm_src, v, bm_dst, vhash); | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm_src, v, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 			/* edge pass */ | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (e, &eiter, f, BM_EDGES_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 				if (!BMO_elem_flag_test(bm_src, e, DUPE_DONE)) { | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 					copy_edge(op, slot_boundary_map_out, bm_src, e, bm_dst, vhash, ehash); | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 					BMO_elem_flag_enable(bm_src, e, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-19 10:39:32 +00:00
										 |  |  | 			copy_face(op, slot_face_map_out, bm_src, f, bm_dst, vhash, ehash); | 
					
						
							| 
									
										
										
										
											2012-11-20 13:29:27 +00:00
										 |  |  | 			BMO_elem_flag_enable(bm_src, f, DUPE_DONE); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* free pointer hashes */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BLI_ghash_free(vhash, NULL, NULL); | 
					
						
							|  |  |  | 	BLI_ghash_free(ehash, NULL, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * Duplicate Operator | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Duplicates verts, edges and faces of a mesh. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * INPUT SLOTS: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * BMOP_DUPE_VINPUT: Buffer containing pointers to mesh vertices to be duplicated | 
					
						
							|  |  |  |  * BMOP_DUPE_EINPUT: Buffer containing pointers to mesh edges to be duplicated | 
					
						
							|  |  |  |  * BMOP_DUPE_FINPUT: Buffer containing pointers to mesh faces to be duplicated | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * OUTPUT SLOTS: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * BMOP_DUPE_VORIGINAL: Buffer containing pointers to the original mesh vertices | 
					
						
							|  |  |  |  * BMOP_DUPE_EORIGINAL: Buffer containing pointers to the original mesh edges | 
					
						
							|  |  |  |  * BMOP_DUPE_FORIGINAL: Buffer containing pointers to the original mesh faces | 
					
						
							|  |  |  |  * BMOP_DUPE_VNEW: Buffer containing pointers to the new mesh vertices | 
					
						
							|  |  |  |  * BMOP_DUPE_ENEW: Buffer containing pointers to the new mesh edges | 
					
						
							|  |  |  |  * BMOP_DUPE_FNEW: Buffer containing pointers to the new mesh faces | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_duplicate_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOperator *dupeop = op; | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMesh *bm2 = BMO_slot_ptr_get(op->slots_in, "dest"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (!bm2) | 
					
						
							|  |  |  | 		bm2 = bm; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-31 12:29:17 +00:00
										 |  |  | 	/* flag input */ | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_flag_enable(bm, dupeop->slots_in, "geom", BM_ALL_NOLOOP, DUPE_INPUT); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* use the internal copy function */ | 
					
						
							| 
									
										
										
										
											2012-11-18 12:14:22 +00:00
										 |  |  | 	bmo_mesh_copy(dupeop, bm, bm2); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-27 11:03:10 +00:00
										 |  |  | 	/* Output */ | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* First copy the input buffers to output buffers - original data */ | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_copy(dupeop, slots_in,  "geom", | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	              dupeop, slots_out, "geom_orig.out"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* Now alloc the new output buffers */ | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, dupeop, dupeop->slots_out, "geom.out", BM_ALL_NOLOOP, DUPE_NEW); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0 /* UNUSED */
 | 
					
						
							|  |  |  | /* executes the duplicate operation, feeding elements of
 | 
					
						
							|  |  |  |  * type flag etypeflag and header flag flag to it.  note, | 
					
						
							|  |  |  |  * to get more useful information (such as the mapping from | 
					
						
							|  |  |  |  * original to new elements) you should run the dupe op manually */ | 
					
						
							| 
									
										
										
										
											2012-03-19 08:36:27 +00:00
										 |  |  | void BMO_dupe_from_flag(BMesh *bm, int htype, const char hflag) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOperator dupeop; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | 	BMO_op_init(bm, &dupeop, "duplicate"); | 
					
						
							| 
									
										
										
										
											2012-03-30 17:30:49 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_hflag(bm, &dupeop, "geom", htype, hflag); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BMO_op_exec(bm, &dupeop); | 
					
						
							|  |  |  | 	BMO_op_finish(bm, &dupeop); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * Split Operator | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Duplicates verts, edges and faces of a mesh but also deletes the originals. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * INPUT SLOTS: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * BMOP_DUPE_VINPUT: Buffer containing pointers to mesh vertices to be split | 
					
						
							|  |  |  |  * BMOP_DUPE_EINPUT: Buffer containing pointers to mesh edges to be split | 
					
						
							|  |  |  |  * BMOP_DUPE_FINPUT: Buffer containing pointers to mesh faces to be split | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * OUTPUT SLOTS: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * BMOP_DUPE_VOUTPUT: Buffer containing pointers to the split mesh vertices | 
					
						
							|  |  |  |  * BMOP_DUPE_EOUTPUT: Buffer containing pointers to the split mesh edges | 
					
						
							|  |  |  |  * BMOP_DUPE_FOUTPUT: Buffer containing pointers to the split mesh faces | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_split_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-30 08:54:50 +00:00
										 |  |  | #define SPLIT_INPUT 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMOperator *splitop = op; | 
					
						
							|  |  |  | 	BMOperator dupeop; | 
					
						
							|  |  |  | 	BMOperator delop; | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_only_faces = BMO_slot_bool_get(op->slots_in, "use_only_faces"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* initialize our sub-operator */ | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 	BMO_op_init(bm, &dupeop, op->flag, "duplicate"); | 
					
						
							|  |  |  | 	BMO_op_init(bm, &delop, op->flag, "delete"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_copy(splitop, slots_in, "geom", | 
					
						
							|  |  |  | 	              &dupeop, slots_in, "geom"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMO_op_exec(bm, &dupeop); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_flag_enable(bm, splitop->slots_in, "geom", BM_ALL_NOLOOP, SPLIT_INPUT); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 	if (use_only_faces) { | 
					
						
							|  |  |  | 		BMVert *v; | 
					
						
							|  |  |  | 		BMEdge *e; | 
					
						
							|  |  |  | 		BMFace *f; | 
					
						
							|  |  |  | 		BMIter iter, iter2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* make sure to remove edges and verts we don't need */ | 
					
						
							| 
									
										
										
										
											2013-08-01 18:33:35 +00:00
										 |  |  | 		BM_ITER_MESH (e, &iter, bm, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  | 			bool found = false; | 
					
						
							|  |  |  | 			BM_ITER_ELEM (f, &iter2, e, BM_FACES_OF_EDGE) { | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 				if (!BMO_elem_flag_test(bm, f, SPLIT_INPUT)) { | 
					
						
							| 
									
										
										
										
											2013-08-01 18:33:35 +00:00
										 |  |  | 					found = true; | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-08-01 18:33:35 +00:00
										 |  |  | 			if (found == false) { | 
					
						
							| 
									
										
										
										
											2012-02-25 19:43:51 +00:00
										 |  |  | 				BMO_elem_flag_enable(bm, e, SPLIT_INPUT); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-01 18:33:35 +00:00
										 |  |  | 		BM_ITER_MESH (v, &iter, bm,  BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 			bool found = false; | 
					
						
							|  |  |  | 			BM_ITER_ELEM (e, &iter2, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 				if (!BMO_elem_flag_test(bm, e, SPLIT_INPUT)) { | 
					
						
							| 
									
										
										
										
											2013-08-01 18:33:35 +00:00
										 |  |  | 					found = true; | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-08-01 18:33:35 +00:00
										 |  |  | 			if (found == false) { | 
					
						
							| 
									
										
										
										
											2012-02-25 19:43:51 +00:00
										 |  |  | 				BMO_elem_flag_enable(bm, v, SPLIT_INPUT); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-02-23 16:00:53 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 22:39:56 +00:00
										 |  |  | 	/* connect outputs of dupe to delete, exluding keep geometry */ | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_int_set(delop.slots_in, "context", DEL_FACES); | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, &delop, delop.slots_in, "geom", BM_ALL_NOLOOP, SPLIT_INPUT); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	BMO_op_exec(bm, &delop); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* now we make our outputs by copying the dupe output */ | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	BMO_slot_copy(&dupeop, slots_out, "geom.out", | 
					
						
							|  |  |  | 	              splitop, slots_out, "geom.out"); | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_slot_copy(&dupeop, slots_out, "boundary_map.out", | 
					
						
							|  |  |  | 	              splitop, slots_out, "boundary_map.out"); | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_slot_copy(&dupeop, slots_out, "isovert_map.out", | 
					
						
							|  |  |  | 	              splitop, slots_out, "isovert_map.out"); | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-01 20:09:17 +00:00
										 |  |  | 	/* cleanup */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMO_op_finish(bm, &delop); | 
					
						
							|  |  |  | 	BMO_op_finish(bm, &dupeop); | 
					
						
							| 
									
										
										
										
											2013-03-30 08:54:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef SPLIT_INPUT
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_delete_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #define DEL_INPUT 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BMOperator *delop = op; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Mark Buffer */ | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_flag_enable(bm, delop->slots_in, "geom", BM_ALL_NOLOOP, DEL_INPUT); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_remove_tagged_context(bm, DEL_INPUT, BMO_slot_int_get(op->slots_in, "context")); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef DEL_INPUT
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-02 01:57:56 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  * Spin Operator | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Extrude or duplicate geometry a number of times, | 
					
						
							|  |  |  |  * rotating and possibly translating after each step | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_spin_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOperator dupop, extop; | 
					
						
							|  |  |  | 	float cent[3], dvec[3]; | 
					
						
							| 
									
										
										
										
											2012-12-27 01:02:32 +00:00
										 |  |  | 	float axis[3]; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	float rmat[3][3]; | 
					
						
							| 
									
										
										
										
											2012-04-02 06:43:16 +00:00
										 |  |  | 	float phi; | 
					
						
							| 
									
										
										
										
											2012-02-20 01:52:35 +00:00
										 |  |  | 	int steps, do_dupli, a, usedvec; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "cent", cent); | 
					
						
							|  |  |  | 	BMO_slot_vec_get(op->slots_in, "axis", axis); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	normalize_v3(axis); | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "dvec", dvec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	usedvec = !is_zero_v3(dvec); | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	steps    = BMO_slot_int_get(op->slots_in,   "steps"); | 
					
						
							| 
									
										
										
										
											2012-11-28 00:47:33 +00:00
										 |  |  | 	phi      = BMO_slot_float_get(op->slots_in, "angle") / steps; | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	do_dupli = BMO_slot_bool_get(op->slots_in,  "use_duplicate"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-15 15:16:11 +00:00
										 |  |  | 	axis_angle_normalized_to_mat3(rmat, axis, phi); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_copy(op, slots_in,  "geom", | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	              op, slots_out, "geom_last.out"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	for (a = 0; a < steps; a++) { | 
					
						
							| 
									
										
										
										
											2012-02-20 01:52:35 +00:00
										 |  |  | 		if (do_dupli) { | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 			BMO_op_initf(bm, &dupop, op->flag, "duplicate geom=%S", op, "geom_last.out"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMO_op_exec(bm, &dupop); | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 			BMO_op_callf(bm, op->flag, | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 			             "rotate cent=%v matrix=%m3 space=%s verts=%S", | 
					
						
							|  |  |  | 			             cent, rmat, op, "space", &dupop, "geom.out"); | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 			BMO_slot_copy(&dupop, slots_out, "geom.out", | 
					
						
							|  |  |  | 			              op,     slots_out, "geom_last.out"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMO_op_finish(bm, &dupop); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 			BMO_op_initf(bm, &extop, op->flag, "extrude_face_region geom=%S", | 
					
						
							|  |  |  | 			             op, "geom_last.out"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMO_op_exec(bm, &extop); | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 			BMO_op_callf(bm, op->flag, | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 			             "rotate cent=%v matrix=%m3 space=%s verts=%S", | 
					
						
							|  |  |  | 			             cent, rmat, op, "space", &extop, "geom.out"); | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 			BMO_slot_copy(&extop, slots_out, "geom.out", | 
					
						
							|  |  |  | 			              op,     slots_out, "geom_last.out"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMO_op_finish(bm, &extop); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-13 16:06:13 +00:00
										 |  |  | 		if (usedvec) { | 
					
						
							|  |  |  | 			mul_m3_v3(rmat, dvec); | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 			BMO_op_callf(bm, op->flag, | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 			             "translate vec=%v space=%s verts=%S", | 
					
						
							|  |  |  | 			             dvec, op, "space", op, "geom_last.out"); | 
					
						
							| 
									
										
										
										
											2012-06-13 16:06:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } |