| 
									
										
										
										
											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-06-08 23:43:11 +00:00
										 |  |  | /** \file blender/bmesh/operators/bmo_utils.c
 | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  |  *  \ingroup bmesh | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * utility bmesh operators, e.g. transform, | 
					
						
							|  |  |  |  * translate, rotate, scale, etc. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | #include "BLI_array.h"
 | 
					
						
							|  |  |  | #include "BLI_heap.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_customdata.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "bmesh.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-08 03:25:53 +00:00
										 |  |  | #include "intern/bmesh_operators_private.h" /* own include */
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_create_vert_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "co", vec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-29 16:26:39 +00:00
										 |  |  | 	BMO_elem_flag_enable(bm, BM_vert_create(bm, vec, NULL, 0), 1); | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, op, op->slots_out, "vert.out", BM_VERT, 1); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | void bmo_transform_exec(BMesh *UNUSED(bm), BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOIter iter; | 
					
						
							|  |  |  | 	BMVert *v; | 
					
						
							|  |  |  | 	float mat[4][4]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_slot_mat4_get(op->slots_in, "matrix", mat); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (v, &iter, op->slots_in, "verts", BM_VERT) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		mul_m4_v3(mat, v->co); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_translate_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float mat[4][4], vec[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "vec", vec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	unit_m4(mat); | 
					
						
							|  |  |  | 	copy_v3_v3(mat[3], vec); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "transform matrix=%m4 verts=%s", mat, op, "verts"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_scale_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float mat[3][3], vec[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "vec", vec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	unit_m3(mat); | 
					
						
							|  |  |  | 	mat[0][0] = vec[0]; | 
					
						
							|  |  |  | 	mat[1][1] = vec[1]; | 
					
						
							|  |  |  | 	mat[2][2] = vec[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "transform matrix=%m3 verts=%s", mat, op, "verts"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_rotate_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float vec[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "cent", vec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* there has to be a proper matrix way to do this, but
 | 
					
						
							|  |  |  | 	 * this is how editmesh did it and I'm too tired to think | 
					
						
							|  |  |  | 	 * through the math right now. */ | 
					
						
							|  |  |  | 	mul_v3_fl(vec, -1.0f); | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "translate verts=%s vec=%v", op, "verts", vec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "transform matrix=%s verts=%s", op, "matrix", op, "verts"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	mul_v3_fl(vec, -1.0f); | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "translate verts=%s vec=%v", op, "verts", vec); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_reverse_faces_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOIter siter; | 
					
						
							|  |  |  | 	BMFace *f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (f, &siter, op->slots_in, "faces", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		BM_face_normal_flip(bm, f); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_rotate_edges_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOIter siter; | 
					
						
							|  |  |  | 	BMEdge *e, *e2; | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_ccw   = BMO_slot_bool_get(op->slots_in, "use_ccw"); | 
					
						
							|  |  |  | 	const bool is_single = BMO_slot_buffer_count(op->slots_in, "edges") == 1; | 
					
						
							| 
									
										
										
										
											2012-03-05 00:50:18 +00:00
										 |  |  | 	short check_flag = is_single ? | 
					
						
							|  |  |  | 	            BM_EDGEROT_CHECK_EXISTS : | 
					
						
							|  |  |  | 	            BM_EDGEROT_CHECK_EXISTS | BM_EDGEROT_CHECK_DEGENERATE; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | #define EDGE_OUT   1
 | 
					
						
							|  |  |  | #define FACE_TAINT 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (e, &siter, op->slots_in, "edges", BM_EDGE) { | 
					
						
							| 
									
										
										
										
											2012-03-03 12:49:03 +00:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * this ends up being called twice, could add option to not to call check in | 
					
						
							|  |  |  | 		 * #BM_edge_rotate to get some extra speed */ | 
					
						
							| 
									
										
										
										
											2012-04-19 14:38:09 +00:00
										 |  |  | 		if (BM_edge_rotate_check(e)) { | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 			BMFace *fa, *fb; | 
					
						
							|  |  |  | 			if (BM_edge_face_pair(e, &fa, &fb)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* check we're untouched */ | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 				if (BMO_elem_flag_test(bm, fa, FACE_TAINT) == false && | 
					
						
							|  |  |  | 				    BMO_elem_flag_test(bm, fb, FACE_TAINT) == false) | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 				{ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 					if (!(e2 = BM_edge_rotate(bm, e, use_ccw, check_flag))) { | 
					
						
							| 
									
										
										
										
											2012-03-05 00:50:18 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 						BMO_error_raise(bm, op, BMERR_INVALID_SELECTION, "Could not rotate edge"); | 
					
						
							|  |  |  | 						return; | 
					
						
							| 
									
										
										
										
											2012-03-05 00:50:18 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 						continue; | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, e2, EDGE_OUT); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-18 07:38:51 +00:00
										 |  |  | 					/* don't touch again */ | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 					BMO_elem_flag_enable(bm, fa, FACE_TAINT); | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, fb, FACE_TAINT); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, op, op->slots_out, "edges.out", BM_EDGE, EDGE_OUT); | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef EDGE_OUT
 | 
					
						
							|  |  |  | #undef FACE_TAINT
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define SEL_FLAG	1
 | 
					
						
							|  |  |  | #define SEL_ORIG	2
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | static void bmo_region_extend_extend(BMesh *bm, BMOperator *op, const bool use_faces) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMVert *v; | 
					
						
							|  |  |  | 	BMEdge *e; | 
					
						
							|  |  |  | 	BMIter eiter; | 
					
						
							|  |  |  | 	BMOIter siter; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	if (!use_faces) { | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 		BMO_ITER (v, &siter, op->slots_in, "geom", BM_VERT) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (e, &eiter, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-08-22 15:10:07 +00:00
										 |  |  | 				if (!BM_elem_flag_test(e, BM_ELEM_HIDDEN)) | 
					
						
							|  |  |  | 					if (!BMO_elem_flag_test(bm, e, SEL_ORIG)) | 
					
						
							|  |  |  | 						break; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (e) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (e, &eiter, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-08-22 15:10:07 +00:00
										 |  |  | 					if (!BM_elem_flag_test(e, BM_ELEM_HIDDEN)) { | 
					
						
							|  |  |  | 						BMO_elem_flag_enable(bm, e, SEL_FLAG); | 
					
						
							|  |  |  | 						BMO_elem_flag_enable(bm, BM_edge_other_vert(e, v), SEL_FLAG); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		BMIter liter, fiter; | 
					
						
							|  |  |  | 		BMFace *f, *f2; | 
					
						
							|  |  |  | 		BMLoop *l; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 		BMO_ITER (f, &siter, op->slots_in, "geom", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (l, &liter, f, BM_LOOPS_OF_FACE) { | 
					
						
							|  |  |  | 				BM_ITER_ELEM (f2, &fiter, l->e, BM_FACES_OF_EDGE) { | 
					
						
							| 
									
										
										
										
											2012-08-22 15:10:07 +00:00
										 |  |  | 					if (!BM_elem_flag_test(f2, BM_ELEM_HIDDEN)) { | 
					
						
							|  |  |  | 						if (!BMO_elem_flag_test(bm, f2, SEL_ORIG)) { | 
					
						
							|  |  |  | 							BMO_elem_flag_enable(bm, f2, SEL_FLAG); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2012-02-25 19:43:51 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | static void bmo_region_extend_constrict(BMesh *bm, BMOperator *op, const bool use_faces) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMVert *v; | 
					
						
							|  |  |  | 	BMEdge *e; | 
					
						
							|  |  |  | 	BMIter eiter; | 
					
						
							|  |  |  | 	BMOIter siter; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	if (!use_faces) { | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 		BMO_ITER (v, &siter, op->slots_in, "geom", BM_VERT) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (e, &eiter, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-08-22 15:10:07 +00:00
										 |  |  | 				if (!BM_elem_flag_test(e, BM_ELEM_HIDDEN)) | 
					
						
							|  |  |  | 					if (!BMO_elem_flag_test(bm, e, SEL_ORIG)) | 
					
						
							|  |  |  | 						break; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (e) { | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm, v, SEL_FLAG); | 
					
						
							| 
									
										
										
										
											2012-02-23 15:47:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (e, &eiter, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-08-22 15:10:07 +00:00
										 |  |  | 					if (!BM_elem_flag_test(e, BM_ELEM_HIDDEN)) { | 
					
						
							|  |  |  | 						BMO_elem_flag_enable(bm, e, SEL_FLAG); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-02-23 15:47:18 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		BMIter liter, fiter; | 
					
						
							|  |  |  | 		BMFace *f, *f2; | 
					
						
							|  |  |  | 		BMLoop *l; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 		BMO_ITER (f, &siter, op->slots_in, "geom", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 			BM_ITER_ELEM (l, &liter, f, BM_LOOPS_OF_FACE) { | 
					
						
							|  |  |  | 				BM_ITER_ELEM (f2, &fiter, l->e, BM_FACES_OF_EDGE) { | 
					
						
							| 
									
										
										
										
											2012-08-22 15:10:07 +00:00
										 |  |  | 					if (!BM_elem_flag_test(f2, BM_ELEM_HIDDEN)) { | 
					
						
							|  |  |  | 						if (!BMO_elem_flag_test(bm, f2, SEL_ORIG)) { | 
					
						
							|  |  |  | 							BMO_elem_flag_enable(bm, f, SEL_FLAG); | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_region_extend_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_faces = BMO_slot_bool_get(op->slots_in, "use_faces"); | 
					
						
							|  |  |  | 	const bool constrict = BMO_slot_bool_get(op->slots_in, "use_constrict"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_flag_enable(bm, op->slots_in, "geom", BM_ALL_NOLOOP, SEL_ORIG); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (constrict) | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | 		bmo_region_extend_constrict(bm, op, use_faces); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | 		bmo_region_extend_extend(bm, op, use_faces); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-27 00:50:59 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, op, op->slots_out, "geom.out", BM_ALL_NOLOOP, SEL_FLAG); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /********* righthand faces implementation ****** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define FACE_VIS	1
 | 
					
						
							|  |  |  | #define FACE_FLAG	2
 | 
					
						
							| 
									
										
										
										
											2012-11-08 06:46:10 +00:00
										 |  |  | // #define FACE_MARK	4  /* UNUSED */
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | #define FACE_FLIP	8
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | /* NOTE: these are the original recalc_face_normals comment in editmesh_mods.c,
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  |  *       copied here for reference. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* based at a select-connected to witness loose objects */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* count per edge the amount of faces
 | 
					
						
							|  |  |  |  * find the ultimate left, front, upper face (not manhattan dist!!) | 
					
						
							|  |  |  |  * also evaluate both triangle cases in quad, since these can be non-flat | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * put normal to the outside, and set the first direction flags in edges | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * then check the object, and set directions / direction-flags: but only for edges with 1 or 2 faces | 
					
						
							|  |  |  |  * this is in fact the 'select connected' | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * in case (selected) faces were not done: start over with 'find the ultimate ...' */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* NOTE: this function uses recursion, which is a little unusual for a bmop
 | 
					
						
							|  |  |  |  *       function, but acceptable I think. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* NOTE: BM_ELEM_TAG is used on faces to tell if they are flipped. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_recalc_face_normals_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMIter liter, liter2; | 
					
						
							|  |  |  | 	BMOIter siter; | 
					
						
							| 
									
										
										
										
											2013-05-12 12:23:44 +00:00
										 |  |  | 	BMFace *f, *startf; | 
					
						
							|  |  |  | 	BMFace **fstack = MEM_mallocN(sizeof(*fstack) * BMO_slot_buffer_count(op->slots_in, "faces"), __func__); | 
					
						
							|  |  |  | 	STACK_DECLARE(fstack); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMLoop *l, *l2; | 
					
						
							| 
									
										
										
										
											2012-02-25 20:58:03 +00:00
										 |  |  | 	float maxx, maxx_test, cent[3]; | 
					
						
							| 
									
										
										
										
											2013-04-22 20:15:42 +00:00
										 |  |  | 	const bool use_flip = BMO_slot_bool_get(op->slots_in, "use_face_tag"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	startf = NULL; | 
					
						
							|  |  |  | 	maxx = -1.0e10; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_buffer_flag_enable(bm, op->slots_in, "faces", BM_FACE, FACE_FLAG); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* find a starting face */ | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (f, &siter, op->slots_in, "faces", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* clear dirty flag */ | 
					
						
							|  |  |  | 		BM_elem_flag_disable(f, BM_ELEM_TAG); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(bm, f, FACE_VIS)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!startf) startf = f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-23 01:19:50 +00:00
										 |  |  | 		BM_face_calc_center_bounds(f, cent); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-25 20:58:03 +00:00
										 |  |  | 		if ((maxx_test = dot_v3v3(cent, cent)) > maxx) { | 
					
						
							|  |  |  | 			maxx = maxx_test; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			startf = f; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!startf) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-23 01:19:50 +00:00
										 |  |  | 	BM_face_calc_center_bounds(startf, cent); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* make sure the starting face has the correct winding */ | 
					
						
							|  |  |  | 	if (dot_v3v3(cent, startf->no) < 0.0f) { | 
					
						
							|  |  |  | 		BM_face_normal_flip(bm, startf); | 
					
						
							|  |  |  | 		BMO_elem_flag_toggle(bm, startf, FACE_FLIP); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 		if (use_flip) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BM_elem_flag_toggle(startf, BM_ELEM_TAG); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* now that we've found our starting face, make all connected faces
 | 
					
						
							|  |  |  | 	 * have the same winding.  this is done recursively, using a manual | 
					
						
							|  |  |  | 	 * stack (if we use simple function recursion, we'd end up overloading | 
					
						
							|  |  |  | 	 * the stack on large meshes). */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-13 02:10:59 +00:00
										 |  |  | 	STACK_INIT(fstack); | 
					
						
							| 
									
										
										
										
											2013-05-12 12:23:44 +00:00
										 |  |  | 	STACK_PUSH(fstack, startf); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMO_elem_flag_enable(bm, startf, FACE_VIS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 12:23:44 +00:00
										 |  |  | 	while ((f = STACK_POP(fstack))) { | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 		BM_ITER_ELEM (l, &liter, f, BM_LOOPS_OF_FACE) { | 
					
						
							|  |  |  | 			BM_ITER_ELEM (l2, &liter2, l, BM_LOOPS_OF_LOOP) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				if (!BMO_elem_flag_test(bm, l2->f, FACE_FLAG) || l2 == l) | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (!BMO_elem_flag_test(bm, l2->f, FACE_VIS)) { | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, l2->f, FACE_VIS); | 
					
						
							| 
									
										
										
										
											2013-05-12 12:23:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					if (l2->v == l->v) { | 
					
						
							|  |  |  | 						BM_face_normal_flip(bm, l2->f); | 
					
						
							|  |  |  | 						 | 
					
						
							|  |  |  | 						BMO_elem_flag_toggle(bm, l2->f, FACE_FLIP); | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 						if (use_flip) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 							BM_elem_flag_toggle(l2->f, BM_ELEM_TAG); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else if (BM_elem_flag_test(l2->f, BM_ELEM_TAG) || BM_elem_flag_test(l->f, BM_ELEM_TAG)) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 						if (use_flip) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 							BM_elem_flag_disable(l->f, BM_ELEM_TAG); | 
					
						
							|  |  |  | 							BM_elem_flag_disable(l2->f, BM_ELEM_TAG); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					 | 
					
						
							| 
									
										
										
										
											2013-05-12 12:23:44 +00:00
										 |  |  | 					STACK_PUSH(fstack, l2->f); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 12:23:44 +00:00
										 |  |  | 	MEM_freeN(fstack); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* check if we have faces yet to do.  if so, recurse */ | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (f, &siter, op->slots_in, "faces", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		if (!BMO_elem_flag_test(bm, f, FACE_VIS)) { | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | 			bmo_recalc_face_normals_exec(bm, op); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | void bmo_smooth_vert_exec(BMesh *UNUSED(bm), BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOIter siter; | 
					
						
							|  |  |  | 	BMIter iter; | 
					
						
							|  |  |  | 	BMVert *v; | 
					
						
							|  |  |  | 	BMEdge *e; | 
					
						
							| 
									
										
										
										
											2013-05-12 12:06:08 +00:00
										 |  |  | 	float (*cos)[3] = MEM_mallocN(sizeof(*cos) * BMO_slot_buffer_count(op->slots_in, "verts"), __func__); | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	float *co, *co2, clip_dist = BMO_slot_float_get(op->slots_in, "clip_dist"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	int i, j, clipx, clipy, clipz; | 
					
						
							| 
									
										
										
										
											2012-07-27 17:35:02 +00:00
										 |  |  | 	int xaxis, yaxis, zaxis; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	clipx = BMO_slot_bool_get(op->slots_in, "mirror_clip_x"); | 
					
						
							|  |  |  | 	clipy = BMO_slot_bool_get(op->slots_in, "mirror_clip_y"); | 
					
						
							|  |  |  | 	clipz = BMO_slot_bool_get(op->slots_in, "mirror_clip_z"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	xaxis = BMO_slot_bool_get(op->slots_in, "use_axis_x"); | 
					
						
							|  |  |  | 	yaxis = BMO_slot_bool_get(op->slots_in, "use_axis_y"); | 
					
						
							|  |  |  | 	zaxis = BMO_slot_bool_get(op->slots_in, "use_axis_z"); | 
					
						
							| 
									
										
										
										
											2012-07-27 17:35:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	i = 0; | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (v, &siter, op->slots_in, "verts", BM_VERT) { | 
					
						
							| 
									
										
										
										
											2013-01-20 16:58:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		co = cos[i]; | 
					
						
							| 
									
										
										
										
											2013-01-20 16:58:14 +00:00
										 |  |  | 		zero_v3(co); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		j = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 		BM_ITER_ELEM (e, &iter, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			co2 = BM_edge_other_vert(e, v)->co; | 
					
						
							|  |  |  | 			add_v3_v3v3(co, co, co2); | 
					
						
							|  |  |  | 			j += 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (!j) { | 
					
						
							|  |  |  | 			copy_v3_v3(co, v->co); | 
					
						
							|  |  |  | 			i++; | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		mul_v3_fl(co, 1.0f / (float)j); | 
					
						
							|  |  |  | 		mid_v3_v3v3(co, co, v->co); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 		if (clipx && fabsf(v->co[0]) <= clip_dist) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			co[0] = 0.0f; | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 		if (clipy && fabsf(v->co[1]) <= clip_dist) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			co[1] = 0.0f; | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 		if (clipz && fabsf(v->co[2]) <= clip_dist) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			co[2] = 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i = 0; | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (v, &siter, op->slots_in, "verts", BM_VERT) { | 
					
						
							| 
									
										
										
										
											2012-07-27 17:35:02 +00:00
										 |  |  | 		if (xaxis) | 
					
						
							|  |  |  | 			v->co[0] = cos[i][0]; | 
					
						
							|  |  |  | 		if (yaxis) | 
					
						
							|  |  |  | 			v->co[1] = cos[i][1]; | 
					
						
							|  |  |  | 		if (zaxis) | 
					
						
							|  |  |  | 			v->co[2] = cos[i][2]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 12:06:08 +00:00
										 |  |  | 	MEM_freeN(cos); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | /**************************************************************************** *
 | 
					
						
							|  |  |  |  * Cycle UVs for a face | 
					
						
							|  |  |  |  **************************************************************************** */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_rotate_uvs_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-15 09:11:17 +00:00
										 |  |  | 	BMOIter fs_iter;  /* selected faces iterator */ | 
					
						
							|  |  |  | 	BMFace *fs;       /* current face */ | 
					
						
							|  |  |  | 	BMIter l_iter;    /* iteration loop */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_ccw = BMO_slot_bool_get(op->slots_in, "use_ccw"); | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	const int cd_loop_uv_offset = CustomData_get_offset(&bm->ldata, CD_MLOOPUV); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	if (cd_loop_uv_offset != -1) { | 
					
						
							|  |  |  | 		BMO_ITER (fs, &fs_iter, op->slots_in, "faces", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 			if (use_ccw == false) {  /* same loops direction */ | 
					
						
							| 
									
										
										
										
											2012-12-03 08:11:04 +00:00
										 |  |  | 				BMLoop *lf;	/* current face loops */ | 
					
						
							|  |  |  | 				MLoopUV *f_luv; /* first face loop uv */ | 
					
						
							|  |  |  | 				float p_uv[2];	/* previous uvs */ | 
					
						
							|  |  |  | 				float t_uv[2];	/* tmp uvs */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				int n = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (lf, &l_iter, fs, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					/* current loop uv is the previous loop uv */ | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 					MLoopUV *luv = BM_ELEM_CD_GET_VOID_P(lf, cd_loop_uv_offset); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					if (n == 0) { | 
					
						
							|  |  |  | 						f_luv = luv; | 
					
						
							|  |  |  | 						copy_v2_v2(p_uv, luv->uv); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						copy_v2_v2(t_uv, luv->uv); | 
					
						
							|  |  |  | 						copy_v2_v2(luv->uv, p_uv); | 
					
						
							|  |  |  | 						copy_v2_v2(p_uv, t_uv); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					n++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				copy_v2_v2(f_luv->uv, p_uv); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 			else { /* counter loop direction */ | 
					
						
							| 
									
										
										
										
											2012-12-03 08:11:04 +00:00
										 |  |  | 				BMLoop *lf;	/* current face loops */ | 
					
						
							|  |  |  | 				MLoopUV *p_luv; /* previous loop uv */ | 
					
						
							|  |  |  | 				MLoopUV *luv; | 
					
						
							|  |  |  | 				float t_uv[2];	/* current uvs */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				int n = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (lf, &l_iter, fs, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					/* previous loop uv is the current loop uv */ | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 					luv = BM_ELEM_CD_GET_VOID_P(lf, cd_loop_uv_offset); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					if (n == 0) { | 
					
						
							|  |  |  | 						p_luv = luv; | 
					
						
							|  |  |  | 						copy_v2_v2(t_uv, luv->uv); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						copy_v2_v2(p_luv->uv, luv->uv); | 
					
						
							|  |  |  | 						p_luv = luv; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					n++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				copy_v2_v2(luv->uv, t_uv); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | /**************************************************************************** *
 | 
					
						
							|  |  |  |  * Reverse UVs for a face | 
					
						
							|  |  |  |  **************************************************************************** */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | static void bm_face_reverse_uvs(BMFace *f, const int cd_loop_uv_offset) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	BMIter iter; | 
					
						
							|  |  |  | 	BMLoop *l; | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	float (*uvs)[2] = BLI_array_alloca(uvs, f->len); | 
					
						
							| 
									
										
										
										
											2012-04-06 11:50:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	BM_ITER_ELEM_INDEX (l, &iter, f, BM_LOOPS_OF_FACE, i) { | 
					
						
							|  |  |  | 		MLoopUV *luv = BM_ELEM_CD_GET_VOID_P(l, cd_loop_uv_offset); | 
					
						
							|  |  |  | 		copy_v2_v2(uvs[i], luv->uv); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	/* now that we have the uvs in the array, reverse! */ | 
					
						
							|  |  |  | 	BM_ITER_ELEM_INDEX (l, &iter, f, BM_LOOPS_OF_FACE, i) { | 
					
						
							|  |  |  | 		/* current loop uv is the previous loop uv */ | 
					
						
							|  |  |  | 		MLoopUV *luv = BM_ELEM_CD_GET_VOID_P(l, cd_loop_uv_offset); | 
					
						
							|  |  |  | 		copy_v2_v2(luv->uv, uvs[(f->len - i - 1)]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | void bmo_reverse_uvs_exec(BMesh *bm, BMOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMOIter iter; | 
					
						
							|  |  |  | 	BMFace *f; | 
					
						
							|  |  |  | 	const int cd_loop_uv_offset = CustomData_get_offset(&bm->ldata, CD_MLOOPUV); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	if (cd_loop_uv_offset != -1) { | 
					
						
							|  |  |  | 		BMO_ITER (f, &iter, op->slots_in, "faces", BM_FACE) { | 
					
						
							|  |  |  | 			bm_face_reverse_uvs(f, cd_loop_uv_offset); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | /**************************************************************************** *
 | 
					
						
							|  |  |  |  * Cycle colors for a face | 
					
						
							|  |  |  |  **************************************************************************** */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_rotate_colors_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-15 09:11:17 +00:00
										 |  |  | 	BMOIter fs_iter;  /* selected faces iterator */ | 
					
						
							|  |  |  | 	BMFace *fs;       /* current face */ | 
					
						
							|  |  |  | 	BMIter l_iter;    /* iteration loop */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_ccw = BMO_slot_bool_get(op->slots_in, "use_ccw"); | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	const int cd_loop_color_offset = CustomData_get_offset(&bm->ldata, CD_MLOOPCOL); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	if (cd_loop_color_offset != -1) { | 
					
						
							|  |  |  | 		BMO_ITER (fs, &fs_iter, op->slots_in, "faces", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 			if (use_ccw == false) {  /* same loops direction */ | 
					
						
							| 
									
										
										
										
											2012-12-03 08:11:04 +00:00
										 |  |  | 				BMLoop *lf;	/* current face loops */ | 
					
						
							|  |  |  | 				MLoopCol *f_lcol; /* first face loop color */ | 
					
						
							|  |  |  | 				MLoopCol p_col;	/* previous color */ | 
					
						
							|  |  |  | 				MLoopCol t_col;	/* tmp color */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				int n = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (lf, &l_iter, fs, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					/* current loop color is the previous loop color */ | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 					MLoopCol *lcol = BM_ELEM_CD_GET_VOID_P(lf, cd_loop_color_offset); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					if (n == 0) { | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 						f_lcol = lcol; | 
					
						
							|  |  |  | 						p_col = *lcol; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 						t_col = *lcol; | 
					
						
							|  |  |  | 						*lcol = p_col; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 						p_col = t_col; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					n++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				*f_lcol = p_col; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 			else {  /* counter loop direction */ | 
					
						
							| 
									
										
										
										
											2012-12-03 08:11:04 +00:00
										 |  |  | 				BMLoop *lf;	/* current face loops */ | 
					
						
							|  |  |  | 				MLoopCol *p_lcol; /* previous loop color */ | 
					
						
							|  |  |  | 				MLoopCol *lcol; | 
					
						
							|  |  |  | 				MLoopCol t_col;	/* current color */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				int n = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 				BM_ITER_ELEM (lf, &l_iter, fs, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					/* previous loop color is the current loop color */ | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 					lcol = BM_ELEM_CD_GET_VOID_P(lf, cd_loop_color_offset); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					if (n == 0) { | 
					
						
							|  |  |  | 						p_lcol = lcol; | 
					
						
							|  |  |  | 						t_col = *lcol; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						*p_lcol = *lcol; | 
					
						
							|  |  |  | 						p_lcol = lcol; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					n++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				*lcol = t_col; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | /*************************************************************************** *
 | 
					
						
							|  |  |  |  * Reverse colors for a face | 
					
						
							|  |  |  |  *************************************************************************** */ | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | static void bm_face_reverse_colors(BMFace *f, const int cd_loop_color_offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMIter iter; | 
					
						
							|  |  |  | 	BMLoop *l; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MLoopCol *cols = BLI_array_alloca(cols, f->len); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	BM_ITER_ELEM_INDEX (l, &iter, f, BM_LOOPS_OF_FACE, i) { | 
					
						
							|  |  |  | 		MLoopCol *lcol = BM_ELEM_CD_GET_VOID_P(l, cd_loop_color_offset); | 
					
						
							|  |  |  | 		cols[i] = *lcol; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* now that we have the uvs in the array, reverse! */ | 
					
						
							|  |  |  | 	BM_ITER_ELEM_INDEX (l, &iter, f, BM_LOOPS_OF_FACE, i) { | 
					
						
							|  |  |  | 		/* current loop uv is the previous loop color */ | 
					
						
							|  |  |  | 		MLoopCol *lcol = BM_ELEM_CD_GET_VOID_P(l, cd_loop_color_offset); | 
					
						
							|  |  |  | 		*lcol = cols[(f->len - i - 1)]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_reverse_colors_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	BMOIter iter; | 
					
						
							|  |  |  | 	BMFace *f; | 
					
						
							|  |  |  | 	const int cd_loop_color_offset = CustomData_get_offset(&bm->ldata, CD_MLOOPCOL); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-12 14:45:09 +00:00
										 |  |  | 	if (cd_loop_color_offset != -1) { | 
					
						
							|  |  |  | 		BMO_ITER (f, &iter, op->slots_in, "faces", BM_FACE) { | 
					
						
							|  |  |  | 			bm_face_reverse_colors(f, cd_loop_color_offset); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | /*************************************************************************** *
 | 
					
						
							|  |  |  |  * shortest vertex path select | 
					
						
							|  |  |  |  *************************************************************************** */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 12:44:34 +00:00
										 |  |  | typedef struct ElemNode { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMVert *v;	/* vertex */ | 
					
						
							|  |  |  | 	BMVert *parent;	/* node parent id */ | 
					
						
							|  |  |  | 	float weight;	/* node weight */ | 
					
						
							|  |  |  | 	HeapNode *hn;	/* heap node */ | 
					
						
							| 
									
										
										
										
											2012-03-02 12:44:34 +00:00
										 |  |  | } ElemNode; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-30 07:59:25 +00:00
										 |  |  | #define VERT_MARK	1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-30 15:27:13 +00:00
										 |  |  | void bmo_shortest_path_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMIter v_iter;		/* mesh verts iterator */ | 
					
						
							| 
									
										
										
										
											2012-11-27 09:21:57 +00:00
										 |  |  | 	BMVert *sv, *ev;	/* starting vertex, ending vertex */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMVert *v;		/* mesh vertex */ | 
					
						
							|  |  |  | 	Heap *h = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 12:44:34 +00:00
										 |  |  | 	ElemNode *vert_list = NULL; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int num_total = 0 /*, num_sels = 0 */, i = 0; | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	const int type = BMO_slot_int_get(op->slots_in, "type"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-27 09:41:08 +00:00
										 |  |  | 	sv = BMO_slot_buffer_get_single(BMO_slot_get(op->slots_in, "vert_start")); | 
					
						
							|  |  |  | 	ev = BMO_slot_buffer_get_single(BMO_slot_get(op->slots_in, "vert_end")); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	num_total = BM_mesh_elem_count(bm, BM_VERT); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* allocate memory for the nodes */ | 
					
						
							| 
									
										
										
										
											2012-03-02 12:44:34 +00:00
										 |  |  | 	vert_list = (ElemNode *)MEM_mallocN(sizeof(ElemNode) * num_total, "vertex nodes"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* iterate through all the mesh vertices */ | 
					
						
							|  |  |  | 	/* loop through all the vertices and fill the vertices/indices structure */ | 
					
						
							|  |  |  | 	i = 0; | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 	BM_ITER_MESH (v, &v_iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		vert_list[i].v = v; | 
					
						
							|  |  |  | 		vert_list[i].parent = NULL; | 
					
						
							|  |  |  | 		vert_list[i].weight = FLT_MAX; | 
					
						
							|  |  |  | 		BM_elem_index_set(v, i); /* set_inline */ | 
					
						
							|  |  |  | 		i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	bm->elem_index_dirty &= ~BM_VERT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | 	 * we now have everything we need, start Dijkstra path finding algorithm | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* set the distance/weight of the start vertex to 0 */ | 
					
						
							|  |  |  | 	vert_list[BM_elem_index_get(sv)].weight = 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	h = BLI_heap_new(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < num_total; i++) { | 
					
						
							|  |  |  | 		vert_list[i].hn = BLI_heap_insert(h, vert_list[i].weight, vert_list[i].v); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-21 15:20:53 +00:00
										 |  |  | 	while (!BLI_heap_is_empty(h)) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		BMEdge *e; | 
					
						
							|  |  |  | 		BMIter e_i; | 
					
						
							|  |  |  | 		float v_weight; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* take the vertex with the lowest weight out of the heap */ | 
					
						
							|  |  |  | 		BMVert *v = (BMVert *)BLI_heap_popmin(h); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (vert_list[BM_elem_index_get(v)].weight == FLT_MAX) /* this means that there is no path */ | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		v_weight = vert_list[BM_elem_index_get(v)].weight; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 		BM_ITER_ELEM (e, &e_i, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMVert *u; | 
					
						
							|  |  |  | 			float e_weight = v_weight; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (type == VPATH_SELECT_EDGE_LENGTH) | 
					
						
							|  |  |  | 				e_weight += len_v3v3(e->v1->co, e->v2->co); | 
					
						
							|  |  |  | 			else e_weight += 1.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			u = (e->v1 == v) ? e->v2 : e->v1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (e_weight < vert_list[BM_elem_index_get(u)].weight) { /* is this path shorter ? */ | 
					
						
							|  |  |  | 				/* add it if so */ | 
					
						
							|  |  |  | 				vert_list[BM_elem_index_get(u)].parent = v; | 
					
						
							|  |  |  | 				vert_list[BM_elem_index_get(u)].weight = e_weight; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* we should do a heap update node function!!! :-/ */ | 
					
						
							|  |  |  | 				BLI_heap_remove(h, vert_list[BM_elem_index_get(u)].hn); | 
					
						
							|  |  |  | 				BLI_heap_insert(h, e_weight, u); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* now we trace the path (if it exists) */ | 
					
						
							|  |  |  | 	v = ev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (vert_list[BM_elem_index_get(v)].parent != NULL) { | 
					
						
							|  |  |  | 		BMO_elem_flag_enable(bm, v, VERT_MARK); | 
					
						
							|  |  |  | 		v = vert_list[BM_elem_index_get(v)].parent; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_heap_free(h, NULL); | 
					
						
							|  |  |  | 	MEM_freeN(vert_list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, op, op->slots_out, "verts.out", BM_VERT, VERT_MARK); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } |