| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2013-07-28 10:38:25 +00:00
										 |  |  | #include "BLI_alloca.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-21 05:39:46 +00:00
										 |  |  | #define ELE_NEW 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-21 05:39:46 +00:00
										 |  |  | 	BMO_elem_flag_enable(bm, BM_vert_create(bm, vec, NULL, BM_CREATE_NOP), ELE_NEW); | 
					
						
							|  |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, op, op->slots_out, "vert.out", BM_VERT, ELE_NEW); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 	float mat[4][4], mat_space[4][4], imat_space[4][4]; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-28 00:16:06 +00:00
										 |  |  | 	BMO_slot_mat4_get(op->slots_in, "matrix", mat); | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 	BMO_slot_mat4_get(op->slots_in, "space", mat_space); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!is_zero_m4(mat_space)) { | 
					
						
							|  |  |  | 		invert_m4_m4(imat_space, mat_space); | 
					
						
							|  |  |  | 		mul_serie_m4(mat, imat_space, mat, mat_space, NULL, NULL, NULL, NULL, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "transform matrix=%m4 space=%s verts=%s", mat, op, "space", 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]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "transform matrix=%m3 space=%s verts=%s", mat, op, "space", 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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-26 06:12:49 +00:00
										 |  |  | 	float center[3]; | 
					
						
							|  |  |  | 	float mat[4][4]; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-26 06:12:49 +00:00
										 |  |  | 	BMO_slot_vec_get(op->slots_in, "cent", center); | 
					
						
							|  |  |  | 	BMO_slot_mat4_get(op->slots_in, "matrix", mat); | 
					
						
							| 
									
										
										
										
											2013-07-30 10:58:36 +00:00
										 |  |  | 	transform_pivot_set_m4(mat, center); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-26 11:15:22 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "transform matrix=%m4 space=%s verts=%s", mat, op, "space", op, "verts"); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 				{ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-16 05:24:55 +00:00
										 |  |  | 					/* don't touch again (faces will be freed so run before rotating the edge) */ | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, fa, FACE_TAINT); | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, fb, FACE_TAINT); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-20 05:50:19 +00:00
										 |  |  | 					if (!(e2 = BM_edge_rotate(bm, e, use_ccw, check_flag))) { | 
					
						
							| 
									
										
										
										
											2013-10-16 05:24:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						BMO_elem_flag_disable(bm, fa, FACE_TAINT); | 
					
						
							|  |  |  | 						BMO_elem_flag_disable(bm, fb, FACE_TAINT); | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2013-10-16 05:24:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 00:50:18 +00:00
										 |  |  | 						continue; | 
					
						
							| 
									
										
										
										
											2012-03-03 12:35:37 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, e2, EDGE_OUT); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |