| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Joseph Eagar. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  | /** \file blender/bmesh/operators/bmo_dissolve.c
 | 
					
						
							|  |  |  |  *  \ingroup bmesh | 
					
						
							| 
									
										
										
										
											2013-03-30 08:54:50 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Removes isolated geometry regions without creating holes in the mesh. | 
					
						
							| 
									
										
										
										
											2012-04-06 09:21:19 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_array.h"
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "bmesh.h"
 | 
					
						
							| 
									
										
										
										
											2013-08-23 04:22:07 +00:00
										 |  |  | #include "bmesh_tools.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 22:40:11 +00:00
										 |  |  | #include "intern/bmesh_operators_private.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | /* ***_ISGC: mark for garbage-collection */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 18:37:08 +00:00
										 |  |  | #define FACE_MARK   1
 | 
					
						
							|  |  |  | #define FACE_ORIG   2
 | 
					
						
							|  |  |  | #define FACE_NEW    4
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 18:37:08 +00:00
										 |  |  | #define EDGE_MARK   1
 | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | #define EDGE_TAG    2
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | #define EDGE_ISGC   8
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 18:37:08 +00:00
										 |  |  | #define VERT_MARK   1
 | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | #define VERT_TAG    2
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | #define VERT_ISGC   8
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-08 04:00:06 +00:00
										 |  |  | static bool UNUSED_FUNCTION(check_hole_in_region) (BMesh *bm, BMFace *f) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMWalker regwalker; | 
					
						
							|  |  |  | 	BMIter liter2; | 
					
						
							|  |  |  | 	BMLoop *l2, *l3; | 
					
						
							|  |  |  | 	BMFace *f2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* checks if there are any unmarked boundary edges in the face regio */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BMW_init(®walker, bm, BMW_ISLAND, | 
					
						
							| 
									
										
										
										
											2012-03-29 13:09:07 +00:00
										 |  |  | 	         BMW_MASK_NOP, BMW_MASK_NOP, FACE_MARK, | 
					
						
							| 
									
										
										
										
											2012-04-16 08:04:12 +00:00
										 |  |  | 	         BMW_FLAG_NOP, | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	         BMW_NIL_LAY); | 
					
						
							| 
									
										
										
										
											2012-03-29 13:09:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 09:53:56 +00:00
										 |  |  | 	for (f2 = BMW_begin(®walker, f); f2; f2 = BMW_step(®walker)) { | 
					
						
							|  |  |  | 		BM_ITER_ELEM (l2, &liter2, f2, BM_LOOPS_OF_FACE) { | 
					
						
							| 
									
										
										
										
											2012-03-01 17:13:02 +00:00
										 |  |  | 			l3 = l2->radial_next; | 
					
						
							| 
									
										
										
										
											2012-05-06 18:37:08 +00:00
										 |  |  | 			if (BMO_elem_flag_test(bm, l3->f, FACE_MARK) != | 
					
						
							|  |  |  | 			    BMO_elem_flag_test(bm, l2->f, FACE_MARK)) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			{ | 
					
						
							|  |  |  | 				if (!BMO_elem_flag_test(bm, l2->e, EDGE_MARK)) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 					return false; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	BMW_end(®walker); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | static void bm_face_split(BMesh *bm, const short oflag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMIter iter; | 
					
						
							|  |  |  | 	BMVert *v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BMIter liter; | 
					
						
							|  |  |  | 	BM_ITER_MESH (v, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(bm, v, oflag)) { | 
					
						
							|  |  |  | 			if (BM_vert_edge_count(v) > 2) { | 
					
						
							|  |  |  | 				BMLoop *l; | 
					
						
							|  |  |  | 				BM_ITER_ELEM (l, &liter, v, BM_LOOPS_OF_VERT) { | 
					
						
							|  |  |  | 					if (l->f->len > 3) { | 
					
						
							|  |  |  | 						if (BMO_elem_flag_test(bm, l->next->v, oflag) == 0 && | 
					
						
							|  |  |  | 						    BMO_elem_flag_test(bm, l->prev->v, oflag) == 0) | 
					
						
							|  |  |  | 						{ | 
					
						
							| 
									
										
										
										
											2013-12-24 11:04:03 +11:00
										 |  |  | 							BM_face_split(bm, l->f, l->next, l->prev, NULL, NULL, true); | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_dissolve_faces_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMOIter oiter; | 
					
						
							| 
									
										
										
										
											2013-03-12 05:48:30 +00:00
										 |  |  | 	BMFace *f; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMFace ***regions = NULL; | 
					
						
							|  |  |  | 	BMFace **faces = NULL; | 
					
						
							| 
									
										
										
										
											2013-07-28 09:05:27 +00:00
										 |  |  | 	BLI_array_declare(regions); | 
					
						
							|  |  |  | 	BLI_array_declare(faces); | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 	BMFace *act_face = bm->act_face; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMWalker regwalker; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_verts = BMO_slot_bool_get(op->slots_in, "use_verts"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (use_verts) { | 
					
						
							|  |  |  | 		/* tag verts that start out with only 2 edges,
 | 
					
						
							|  |  |  | 		 * don't remove these later */ | 
					
						
							|  |  |  | 		BMIter viter; | 
					
						
							|  |  |  | 		BMVert *v; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-19 13:47:58 +00:00
										 |  |  | 		BM_ITER_MESH (v, &viter, bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-02-25 19:43:51 +00:00
										 |  |  | 			BMO_elem_flag_set(bm, v, VERT_MARK, (BM_vert_edge_count(v) != 2)); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_slot_buffer_flag_enable(bm, op->slots_in, "faces", BM_FACE, FACE_MARK); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* collect region */ | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (f, &oiter, op->slots_in, "faces", BM_FACE) { | 
					
						
							| 
									
										
										
										
											2013-03-12 05:48:30 +00:00
										 |  |  | 		BMFace *f_iter; | 
					
						
							| 
									
										
										
										
											2012-03-01 13:13:08 +00:00
										 |  |  | 		if (!BMO_elem_flag_test(bm, f, FACE_MARK)) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		BLI_array_empty(faces); | 
					
						
							|  |  |  | 		faces = NULL; /* forces different allocatio */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BMW_init(®walker, bm, BMW_ISLAND, | 
					
						
							| 
									
										
										
										
											2012-03-29 13:09:07 +00:00
										 |  |  | 		         BMW_MASK_NOP, BMW_MASK_NOP, FACE_MARK, | 
					
						
							| 
									
										
										
										
											2012-04-16 10:29:59 +00:00
										 |  |  | 		         BMW_FLAG_NOP, /* no need to check BMW_FLAG_TEST_HIDDEN, faces are already marked by the bmo */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		         BMW_NIL_LAY); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-12 05:48:30 +00:00
										 |  |  | 		for (f_iter = BMW_begin(®walker, f); f_iter; f_iter = BMW_step(®walker)) { | 
					
						
							|  |  |  | 			BLI_array_append(faces, f_iter); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		BMW_end(®walker); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		for (i = 0; i < BLI_array_count(faces); i++) { | 
					
						
							| 
									
										
										
										
											2013-03-12 05:48:30 +00:00
										 |  |  | 			f_iter = faces[i]; | 
					
						
							|  |  |  | 			BMO_elem_flag_disable(bm, f_iter, FACE_MARK); | 
					
						
							|  |  |  | 			BMO_elem_flag_enable(bm, f_iter, FACE_ORIG); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (BMO_error_occurred(bm)) { | 
					
						
							|  |  |  | 			BMO_error_clear(bm); | 
					
						
							|  |  |  | 			BMO_error_raise(bm, op, BMERR_DISSOLVEFACES_FAILED, NULL); | 
					
						
							|  |  |  | 			goto cleanup; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		BLI_array_append(faces, NULL); | 
					
						
							|  |  |  | 		BLI_array_append(regions, faces); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	for (i = 0; i < BLI_array_count(regions); i++) { | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 		BMFace *f_new; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		int tot = 0; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		faces = regions[i]; | 
					
						
							|  |  |  | 		if (!faces[0]) { | 
					
						
							|  |  |  | 			BMO_error_raise(bm, op, BMERR_DISSOLVEFACES_FAILED, | 
					
						
							|  |  |  | 			                "Could not find boundary of dissolve region"); | 
					
						
							|  |  |  | 			goto cleanup; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		while (faces[tot]) | 
					
						
							|  |  |  | 			tot++; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 		f_new = BM_faces_join(bm, faces, tot, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (f_new) { | 
					
						
							|  |  |  | 			/* maintain active face */ | 
					
						
							|  |  |  | 			if (act_face && bm->act_face == NULL) { | 
					
						
							|  |  |  | 				bm->act_face = f_new; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			BMO_error_raise(bm, op, BMERR_DISSOLVEFACES_FAILED, | 
					
						
							|  |  |  | 			                "Could not create merged face"); | 
					
						
							|  |  |  | 			goto cleanup; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* if making the new face failed (e.g. overlapping test)
 | 
					
						
							|  |  |  | 		 * unmark the original faces for deletion */ | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 		BMO_elem_flag_disable(bm, f_new, FACE_ORIG); | 
					
						
							|  |  |  | 		BMO_elem_flag_enable(bm, f_new, FACE_NEW); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-21 00:58:02 +00:00
										 |  |  | 	BMO_op_callf(bm, op->flag, "delete geom=%ff context=%i", FACE_ORIG, DEL_FACES); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (use_verts) { | 
					
						
							|  |  |  | 		BMIter viter; | 
					
						
							| 
									
										
										
										
											2014-06-26 23:29:45 +10:00
										 |  |  | 		BMVert *v, *v_next; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-26 23:29:45 +10:00
										 |  |  | 		BM_ITER_MESH_MUTABLE (v, v_next, &viter, bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			if (BMO_elem_flag_test(bm, v, VERT_MARK)) { | 
					
						
							|  |  |  | 				if (BM_vert_edge_count(v) == 2) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 					BM_vert_collapse_edge(bm, v->e, v, true); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-01 20:09:17 +00:00
										 |  |  | 	if (BMO_error_occurred(bm)) { | 
					
						
							|  |  |  | 		goto cleanup; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											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, "region.out", BM_FACE, FACE_NEW); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | cleanup: | 
					
						
							| 
									
										
										
										
											2012-03-01 20:09:17 +00:00
										 |  |  | 	/* free/cleanup */ | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	for (i = 0; i < BLI_array_count(regions); i++) { | 
					
						
							|  |  |  | 		if (regions[i]) MEM_freeN(regions[i]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_array_free(regions); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_dissolve_edges_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* BMOperator fop; */ | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 	BMFace *act_face = bm->act_face; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	BMOIter eiter; | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 	BMIter iter; | 
					
						
							|  |  |  | 	BMEdge *e, *e_next; | 
					
						
							|  |  |  | 	BMVert *v, *v_next; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool use_verts = BMO_slot_bool_get(op->slots_in, "use_verts"); | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | 	const bool use_face_split = BMO_slot_bool_get(op->slots_in, "use_face_split"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (use_face_split) { | 
					
						
							|  |  |  | 		BMO_slot_buffer_flag_enable(bm, op->slots_in, "edges", BM_EDGE, EDGE_TAG); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 		BM_ITER_MESH (v, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 			BMIter itersub; | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | 			int untag_count = 0; | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 			BM_ITER_ELEM (e, &itersub, v, BM_EDGES_OF_VERT) { | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | 				if (!BMO_elem_flag_test(bm, e, EDGE_TAG)) { | 
					
						
							|  |  |  | 					untag_count++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* check that we have 2 edges remaining after dissolve */ | 
					
						
							|  |  |  | 			if (untag_count <= 2) { | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm, v, VERT_TAG); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		bm_face_split(bm, VERT_TAG); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (use_verts) { | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 		BM_ITER_MESH (v, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-02-25 19:43:51 +00:00
										 |  |  | 			BMO_elem_flag_set(bm, v, VERT_MARK, (BM_vert_edge_count(v) != 2)); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 	/* tag all verts/edges connected to faces */ | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMO_ITER (e, &eiter, op->slots_in, "edges", BM_EDGE) { | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 		BMFace *f_pair[2]; | 
					
						
							|  |  |  | 		if (BM_edge_face_pair(e, &f_pair[0], &f_pair[1])) { | 
					
						
							|  |  |  | 			unsigned int j; | 
					
						
							|  |  |  | 			for (j = 0; j < 2; j++) { | 
					
						
							|  |  |  | 				BMLoop *l_first, *l_iter; | 
					
						
							|  |  |  | 				l_iter = l_first = BM_FACE_FIRST_LOOP(f_pair[j]); | 
					
						
							|  |  |  | 				do { | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, l_iter->v, VERT_ISGC); | 
					
						
							|  |  |  | 					BMO_elem_flag_enable(bm, l_iter->e, EDGE_ISGC); | 
					
						
							|  |  |  | 				} while ((l_iter = l_iter->next) != l_first); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-05 01:53:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 	BMO_ITER (e, &eiter, op->slots_in, "edges", BM_EDGE) { | 
					
						
							|  |  |  | 		BMFace *fa, *fb; | 
					
						
							| 
									
										
										
										
											2012-03-05 01:53:30 +00:00
										 |  |  | 		if (BM_edge_face_pair(e, &fa, &fb)) { | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 			BMFace *f_new; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/* join faces */ | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 			f_new = BM_faces_join_pair(bm, fa, fb, e, false); | 
					
						
							| 
									
										
										
										
											2013-03-12 05:36:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (f_new) { | 
					
						
							|  |  |  | 				/* maintain active face */ | 
					
						
							|  |  |  | 				if (act_face && bm->act_face == NULL) { | 
					
						
							|  |  |  | 					bm->act_face = f_new; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 	/* Cleanup geometry (#BM_faces_join_pair, but it removes geometry we're looping on)
 | 
					
						
							|  |  |  | 	 * so do this in a separate pass instead. */ | 
					
						
							|  |  |  | 	BM_ITER_MESH_MUTABLE (e, e_next, &iter, bm, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  | 		if ((e->l == NULL) && BMO_elem_flag_test(bm, e, EDGE_ISGC)) { | 
					
						
							|  |  |  | 			BM_edge_kill(bm, e); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	BM_ITER_MESH_MUTABLE (v, v_next, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 		if ((v->e == NULL) && BMO_elem_flag_test(bm, v, VERT_ISGC)) { | 
					
						
							|  |  |  | 			BM_vert_kill(bm, v); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* done with cleanup */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	if (use_verts) { | 
					
						
							| 
									
										
										
										
											2013-10-03 15:29:08 +00:00
										 |  |  | 		BM_ITER_MESH_MUTABLE (v, v_next, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			if (BMO_elem_flag_test(bm, v, VERT_MARK)) { | 
					
						
							|  |  |  | 				if (BM_vert_edge_count(v) == 2) { | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 					BM_vert_collapse_edge(bm, v->e, v, true); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_dissolve_verts_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	BMOIter oiter; | 
					
						
							|  |  |  | 	BMIter iter; | 
					
						
							| 
									
										
										
										
											2013-10-03 07:06:56 +00:00
										 |  |  | 	BMVert *v, *v_next; | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	BMEdge *e, *e_next; | 
					
						
							|  |  |  | 	BMFace *act_face = bm->act_face; | 
					
						
							| 
									
										
										
										
											2013-05-08 14:01:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	const bool use_face_split = BMO_slot_bool_get(op->slots_in, "use_face_split"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	BMO_ITER (v, &oiter, op->slots_in, "verts", BM_VERT) { | 
					
						
							|  |  |  | 		BMIter itersub; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BMO_elem_flag_enable(bm, v, VERT_MARK | VERT_ISGC); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BM_ITER_ELEM (e, &itersub, v, BM_EDGES_OF_VERT) { | 
					
						
							|  |  |  | 			BMO_elem_flag_enable(bm, e, EDGE_ISGC); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-02 12:44:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-08 14:01:38 +00:00
										 |  |  | 	if (use_face_split) { | 
					
						
							| 
									
										
										
										
											2013-06-14 02:30:40 +00:00
										 |  |  | 		bm_face_split(bm, VERT_MARK); | 
					
						
							| 
									
										
										
										
											2013-05-08 14:01:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	BMO_ITER (v, &oiter, op->slots_in, "verts", BM_VERT) { | 
					
						
							|  |  |  | 		BMIter itersub; | 
					
						
							|  |  |  | 		BMLoop *l_first; | 
					
						
							|  |  |  | 		BM_ITER_ELEM (l_first, &itersub, v, BM_LOOPS_OF_VERT) { | 
					
						
							|  |  |  | 			BMLoop *l_iter; | 
					
						
							|  |  |  | 			l_iter = l_first; | 
					
						
							|  |  |  | 			do { | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm, l_iter->v, VERT_ISGC); | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm, l_iter->e, EDGE_ISGC); | 
					
						
							|  |  |  | 			} while ((l_iter = l_iter->next) != l_first); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	BMO_ITER (v, &oiter, op->slots_in, "verts", BM_VERT) { | 
					
						
							|  |  |  | 		BMIter itersub; | 
					
						
							| 
									
										
										
										
											2012-02-26 05:48:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 		if (BM_vert_edge_count(v) != 2) { | 
					
						
							|  |  |  | 			BM_ITER_ELEM (e, &itersub, v, BM_EDGES_OF_VERT) { | 
					
						
							|  |  |  | 				BMFace *fa, *fb; | 
					
						
							|  |  |  | 				if (BM_edge_face_pair(e, &fa, &fb)) { | 
					
						
							|  |  |  | 					BMFace *f_new; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 					/* join faces */ | 
					
						
							|  |  |  | 					f_new = BM_faces_join_pair(bm, fa, fb, e, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					/* maintain active face */ | 
					
						
							|  |  |  | 					if (act_face && bm->act_face == NULL) { | 
					
						
							|  |  |  | 						bm->act_face = f_new; | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	/* Cleanup geometry (#BM_faces_join_pair, but it removes geometry we're looping on)
 | 
					
						
							|  |  |  | 	 * so do this in a separate pass instead. */ | 
					
						
							|  |  |  | 	BM_ITER_MESH_MUTABLE (e, e_next, &iter, bm, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  | 		if ((e->l == NULL) && BMO_elem_flag_test(bm, e, EDGE_ISGC)) { | 
					
						
							|  |  |  | 			BM_edge_kill(bm, e); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 	BM_ITER_MESH_MUTABLE (v, v_next, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 		if ((v->e == NULL) && BMO_elem_flag_test(bm, v, VERT_ISGC)) { | 
					
						
							|  |  |  | 			BM_vert_kill(bm, v); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* done with cleanup */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BM_ITER_MESH_MUTABLE (v, v_next, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 		if (BMO_elem_flag_test(bm, v, VERT_MARK)) { | 
					
						
							| 
									
										
										
										
											2014-06-27 05:11:02 +10:00
										 |  |  | 			if (BM_vert_edge_count(v) == 2) { | 
					
						
							|  |  |  | 				BM_vert_collapse_edge(bm, v->e, v, true); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-05 01:20:32 +00:00
										 |  |  | /* Limited Dissolve */ | 
					
						
							| 
									
										
										
										
											2012-02-28 09:48:00 +00:00
										 |  |  | void bmo_dissolve_limit_exec(BMesh *bm, BMOperator *op) | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	BMOpSlot *einput = BMO_slot_get(op->slots_in, "edges"); | 
					
						
							|  |  |  | 	BMOpSlot *vinput = BMO_slot_get(op->slots_in, "verts"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 	const float angle_max = (float)M_PI / 2.0f; | 
					
						
							| 
									
										
										
										
											2012-11-19 14:58:31 +00:00
										 |  |  | 	const float angle_limit = min_ff(angle_max, BMO_slot_float_get(op->slots_in, "angle_limit")); | 
					
						
							| 
									
										
										
										
											2013-01-14 16:42:43 +00:00
										 |  |  | 	const bool do_dissolve_boundaries = BMO_slot_bool_get(op->slots_in, "use_dissolve_boundaries"); | 
					
						
							| 
									
										
										
										
											2013-06-03 05:07:16 +00:00
										 |  |  | 	const BMO_Delimit delimit = BMO_slot_int_get(op->slots_in, "delimit"); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-03 05:07:16 +00:00
										 |  |  | 	BM_mesh_decimate_dissolve_ex(bm, angle_limit, do_dissolve_boundaries, delimit, | 
					
						
							| 
									
										
										
										
											2012-11-20 03:29:12 +00:00
										 |  |  | 	                             (BMVert **)BMO_SLOT_AS_BUFFER(vinput), vinput->len, | 
					
						
							| 
									
										
										
										
											2013-07-25 06:05:44 +00:00
										 |  |  | 	                             (BMEdge **)BMO_SLOT_AS_BUFFER(einput), einput->len, | 
					
						
							|  |  |  | 	                             FACE_NEW); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BMO_slot_buffer_from_enabled_flag(bm, op, op->slots_out, "region.out", BM_FACE, FACE_NEW); | 
					
						
							| 
									
										
										
										
											2012-02-19 18:31:04 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-17 11:32:35 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define EDGE_MARK 1
 | 
					
						
							|  |  |  | #define EDGE_COLLAPSE 2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void bm_mesh_edge_collapse_flagged(BMesh *bm, const int flag, const short oflag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMO_op_callf(bm, flag, "collapse edges=%fe", oflag); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void bmo_dissolve_degenerate_exec(BMesh *bm, BMOperator *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const float dist = BMO_slot_float_get(op->slots_in, "dist"); | 
					
						
							|  |  |  | 	const float dist_sq = dist * dist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool found; | 
					
						
							|  |  |  | 	BMIter eiter; | 
					
						
							|  |  |  | 	BMEdge *e; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BMO_slot_buffer_flag_enable(bm, op->slots_in, "edges", BM_EDGE, EDGE_MARK); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* collapse zero length edges, this accounts for zero area faces too */ | 
					
						
							|  |  |  | 	found = false; | 
					
						
							|  |  |  | 	BM_ITER_MESH (e, &eiter, bm, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  | 		if (BMO_elem_flag_test(bm, e, EDGE_MARK)) { | 
					
						
							|  |  |  | 			if (BM_edge_calc_length_squared(e) < dist_sq) { | 
					
						
							|  |  |  | 				BMO_elem_flag_enable(bm, e, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 				found = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* clear all loop tags (checked later) */ | 
					
						
							|  |  |  | 		if (e->l) { | 
					
						
							|  |  |  | 			BMLoop *l_iter, *l_first; | 
					
						
							|  |  |  | 			l_iter = l_first = e->l; | 
					
						
							|  |  |  | 			do { | 
					
						
							|  |  |  | 				BM_elem_flag_disable(l_iter, BM_ELEM_TAG); | 
					
						
							|  |  |  | 			} while ((l_iter = l_iter->radial_next) != l_first); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (found) { | 
					
						
							|  |  |  | 		bm_mesh_edge_collapse_flagged(bm, op->flag, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* clip degenerate ears from the face */ | 
					
						
							|  |  |  | 	found = false; | 
					
						
							|  |  |  | 	BM_ITER_MESH (e, &eiter, bm, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  | 		if (e->l && BMO_elem_flag_test(bm, e, EDGE_MARK)) { | 
					
						
							|  |  |  | 			BMLoop *l_iter, *l_first; | 
					
						
							|  |  |  | 			l_iter = l_first = e->l; | 
					
						
							|  |  |  | 			do { | 
					
						
							|  |  |  | 				if ( | 
					
						
							|  |  |  | 				    /* check the loop hasn't already been tested (and flag not to test again) */ | 
					
						
							|  |  |  | 				    !BM_elem_flag_test(l_iter, BM_ELEM_TAG) && | 
					
						
							|  |  |  | 				    (BM_elem_flag_enable(l_iter, BM_ELEM_TAG), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				     /* check we're marked to tested (radial edge already tested) */ | 
					
						
							|  |  |  | 				     BMO_elem_flag_test(bm, l_iter->prev->e, EDGE_MARK) && | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				     /* check edges are not already going to be collapsed */ | 
					
						
							|  |  |  | 				     !BMO_elem_flag_test(bm, l_iter->e, EDGE_COLLAPSE) && | 
					
						
							|  |  |  | 				     !BMO_elem_flag_test(bm, l_iter->prev->e, EDGE_COLLAPSE))) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					/* test if the faces loop (ear) is degenerate */ | 
					
						
							|  |  |  | 					float dir_prev[3], len_prev; | 
					
						
							|  |  |  | 					float dir_next[3], len_next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					sub_v3_v3v3(dir_prev, l_iter->prev->v->co, l_iter->v->co); | 
					
						
							|  |  |  | 					sub_v3_v3v3(dir_next, l_iter->next->v->co, l_iter->v->co); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					len_prev = normalize_v3(dir_prev); | 
					
						
							|  |  |  | 					len_next = normalize_v3(dir_next); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if ((len_v3v3(dir_prev, dir_next) * min_ff(len_prev, len_next)) <= dist) { | 
					
						
							|  |  |  | 						bool reset = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if (fabsf(len_prev - len_next) <= dist) { | 
					
						
							|  |  |  | 							/* both edges the same length */ | 
					
						
							|  |  |  | 							if (l_iter->f->len == 3) { | 
					
						
							|  |  |  | 								/* ideally this would have been discovered with short edge test above */ | 
					
						
							|  |  |  | 								BMO_elem_flag_enable(bm, l_iter->next->e, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 								found = true; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								/* add a joining edge and tag for removal */ | 
					
						
							|  |  |  | 								BMLoop *l_split; | 
					
						
							|  |  |  | 								if (BM_face_split(bm, l_iter->f, l_iter->prev, l_iter->next, &l_split, NULL, true)) { | 
					
						
							|  |  |  | 									BMO_elem_flag_enable(bm, l_split->e, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 									found = true; | 
					
						
							|  |  |  | 									reset = true; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						else if (len_prev < len_next) { | 
					
						
							|  |  |  | 							/* split 'l_iter->e', then join the vert with next */ | 
					
						
							|  |  |  | 							BMVert *v_new; | 
					
						
							|  |  |  | 							BMEdge *e_new; | 
					
						
							|  |  |  | 							BMLoop *l_split; | 
					
						
							|  |  |  | 							v_new = BM_edge_split(bm, l_iter->e, l_iter->v, &e_new, len_prev / len_next); | 
					
						
							|  |  |  | 							BLI_assert(v_new == l_iter->next->v); | 
					
						
							|  |  |  | 							(void)v_new; | 
					
						
							|  |  |  | 							if (BM_face_split(bm, l_iter->f, l_iter->prev, l_iter->next, &l_split, NULL, true)) { | 
					
						
							|  |  |  | 								BMO_elem_flag_enable(bm, l_split->e, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 								found = true; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							reset = true; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						else if (len_next < len_prev) { | 
					
						
							|  |  |  | 							/* split 'l_iter->prev->e', then join the vert with next */ | 
					
						
							|  |  |  | 							BMVert *v_new; | 
					
						
							|  |  |  | 							BMEdge *e_new; | 
					
						
							|  |  |  | 							BMLoop *l_split; | 
					
						
							|  |  |  | 							v_new = BM_edge_split(bm, l_iter->prev->e, l_iter->v, &e_new, len_next / len_prev); | 
					
						
							|  |  |  | 							BLI_assert(v_new == l_iter->prev->v); | 
					
						
							|  |  |  | 							(void)v_new; | 
					
						
							|  |  |  | 							if (BM_face_split(bm, l_iter->f, l_iter->prev, l_iter->next, &l_split, NULL, true)) { | 
					
						
							|  |  |  | 								BMO_elem_flag_enable(bm, l_split->e, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 								found = true; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							reset = true; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if (reset) { | 
					
						
							|  |  |  | 							/* we can't easily track where we are on the radial edge, reset! */ | 
					
						
							|  |  |  | 							l_first = l_iter; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} while ((l_iter = l_iter->radial_next) != l_first); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (found) { | 
					
						
							|  |  |  | 		bm_mesh_edge_collapse_flagged(bm, op->flag, EDGE_COLLAPSE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |