| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bmesh | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * BMesh decimator that dissolves flat areas into polygons (ngons). | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  | #include "BLI_heap.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  | #include "BKE_customdata.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | #include "bmesh.h"
 | 
					
						
							|  |  |  | #include "bmesh_decimate.h" /* own include */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  | /* check that collapsing a vertex between 2 edges doesn't cause a degenerate face. */ | 
					
						
							|  |  |  | #define USE_DEGENERATE_CHECK
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  | #define COST_INVALID FLT_MAX
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  | struct DelimitData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool bm_edge_is_delimiter(const BMEdge *e, | 
					
						
							|  |  |  |                                  const BMO_Delimit delimit, | 
					
						
							|  |  |  |                                  const struct DelimitData *delimit_data); | 
					
						
							|  |  |  | static bool bm_vert_is_delimiter(const BMVert *v, | 
					
						
							|  |  |  |                                  const BMO_Delimit delimit, | 
					
						
							|  |  |  |                                  const struct DelimitData *delimit_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | /* multiply vertex edge angle by face angle
 | 
					
						
							|  |  |  |  * this means we are not left with sharp corners between _almost_ planer faces | 
					
						
							|  |  |  |  * convert angles [0-PI/2] -> [0-1], multiply together, then convert back to radians. */ | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  | static float bm_vert_edge_face_angle(BMVert *v, | 
					
						
							|  |  |  |                                      const BMO_Delimit delimit, | 
					
						
							|  |  |  |                                      const struct DelimitData *delimit_data) | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  | #define UNIT_TO_ANGLE DEG2RADF(90.0f)
 | 
					
						
							|  |  |  | #define ANGLE_TO_UNIT (1.0f / UNIT_TO_ANGLE)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |   const float angle = BM_vert_calc_edge_angle(v); | 
					
						
							|  |  |  |   /* note: could be either edge, it doesn't matter */ | 
					
						
							|  |  |  |   if (v->e && BM_edge_is_manifold(v->e)) { | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |     /* Checking delimited is important here,
 | 
					
						
							|  |  |  |      * otherwise the boundary between two materials for e.g. | 
					
						
							|  |  |  |      * will collapse if the faces on either side of the edge have a small angle. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * This way, delimiting edges are treated like boundary edges, | 
					
						
							|  |  |  |      * the detail between two delimiting regions won't over-collapse. */ | 
					
						
							|  |  |  |     if (!bm_vert_is_delimiter(v, delimit, delimit_data)) { | 
					
						
							|  |  |  |       return ((angle * ANGLE_TO_UNIT) * (BM_edge_calc_face_angle(v->e) * ANGLE_TO_UNIT)) * | 
					
						
							|  |  |  |              UNIT_TO_ANGLE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |   return angle; | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef UNIT_TO_ANGLE
 | 
					
						
							|  |  |  | #undef ANGLE_TO_UNIT
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  | struct DelimitData { | 
					
						
							|  |  |  |   int cd_loop_type; | 
					
						
							|  |  |  |   int cd_loop_size; | 
					
						
							|  |  |  |   int cd_loop_offset; | 
					
						
							|  |  |  |   int cd_loop_offset_end; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool bm_edge_is_contiguous_loop_cd_all(const BMEdge *e, | 
					
						
							|  |  |  |                                               const struct DelimitData *delimit_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int cd_loop_offset; | 
					
						
							|  |  |  |   for (cd_loop_offset = delimit_data->cd_loop_offset; | 
					
						
							|  |  |  |        cd_loop_offset < delimit_data->cd_loop_offset_end; | 
					
						
							|  |  |  |        cd_loop_offset += delimit_data->cd_loop_size) { | 
					
						
							|  |  |  |     if (BM_edge_is_contiguous_loop_cd(e, delimit_data->cd_loop_type, cd_loop_offset) == false) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  | static bool bm_edge_is_delimiter(const BMEdge *e, | 
					
						
							|  |  |  |                                  const BMO_Delimit delimit, | 
					
						
							|  |  |  |                                  const struct DelimitData *delimit_data) | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |   /* Caller must ensure. */ | 
					
						
							|  |  |  |   BLI_assert(BM_edge_is_manifold(e)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |   if (delimit != 0) { | 
					
						
							|  |  |  |     if (delimit & BMO_DELIM_SEAM) { | 
					
						
							|  |  |  |       if (BM_elem_flag_test(e, BM_ELEM_SEAM)) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (delimit & BMO_DELIM_SHARP) { | 
					
						
							|  |  |  |       if (BM_elem_flag_test(e, BM_ELEM_SMOOTH) == 0) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (delimit & BMO_DELIM_MATERIAL) { | 
					
						
							|  |  |  |       if (e->l->f->mat_nr != e->l->radial_next->f->mat_nr) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (delimit & BMO_DELIM_NORMAL) { | 
					
						
							|  |  |  |       if (!BM_edge_is_contiguous(e)) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (delimit & BMO_DELIM_UV) { | 
					
						
							|  |  |  |       if (bm_edge_is_contiguous_loop_cd_all(e, delimit_data) == 0) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  | static bool bm_vert_is_delimiter(const BMVert *v, | 
					
						
							|  |  |  |                                  const BMO_Delimit delimit, | 
					
						
							|  |  |  |                                  const struct DelimitData *delimit_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_assert(v->e != NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |   if (delimit != 0) { | 
					
						
							|  |  |  |     const BMEdge *e, *e_first; | 
					
						
							|  |  |  |     e = e_first = v->e; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |       if (BM_edge_is_manifold(e)) { | 
					
						
							|  |  |  |         if (bm_edge_is_delimiter(e, delimit, delimit_data)) { | 
					
						
							|  |  |  |           return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } while ((e = BM_DISK_EDGE_NEXT(e, v)) != e_first); | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  | static float bm_edge_calc_dissolve_error(const BMEdge *e, | 
					
						
							|  |  |  |                                          const BMO_Delimit delimit, | 
					
						
							|  |  |  |                                          const struct DelimitData *delimit_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (BM_edge_is_manifold(e) && !bm_edge_is_delimiter(e, delimit, delimit_data)) { | 
					
						
							|  |  |  |     float angle_cos_neg = dot_v3v3(e->l->f->no, e->l->radial_next->f->no); | 
					
						
							|  |  |  |     if (BM_edge_is_contiguous(e)) { | 
					
						
							|  |  |  |       angle_cos_neg *= -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return angle_cos_neg; | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   return COST_INVALID; | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  | #ifdef USE_DEGENERATE_CHECK
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-07 22:56:13 +10:00
										 |  |  | static void mul_v2_m3v3_center(float r[2], | 
					
						
							|  |  |  |                                const float m[3][3], | 
					
						
							|  |  |  |                                const float a[3], | 
					
						
							|  |  |  |                                const float center[3]) | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  | { | 
					
						
							|  |  |  |   BLI_assert(r != a); | 
					
						
							|  |  |  |   BLI_assert(r != center); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   float co[3]; | 
					
						
							|  |  |  |   sub_v3_v3v3(co, a, center); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   r[0] = m[0][0] * co[0] + m[1][0] * co[1] + m[2][0] * co[2]; | 
					
						
							|  |  |  |   r[1] = m[0][1] * co[0] + m[1][1] * co[1] + m[2][1] * co[2]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool bm_loop_collapse_is_degenerate(BMLoop *l_ear) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-02-05 16:23:34 +11:00
										 |  |  |   /* Calculate relative to the central vertex for higher precision. */ | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   const float *center = l_ear->v->co; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   float tri_2d[3][2]; | 
					
						
							|  |  |  |   float axis_mat[3][3]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   axis_dominant_v3_to_m3(axis_mat, l_ear->f->no); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   { | 
					
						
							|  |  |  |     mul_v2_m3v3_center(tri_2d[0], axis_mat, l_ear->prev->v->co, center); | 
					
						
							|  |  |  | #  if 0
 | 
					
						
							|  |  |  |     mul_v2_m3v3_center(tri_2d[1], axis_mat, l_ear->v->co, center); | 
					
						
							|  |  |  | #  else
 | 
					
						
							|  |  |  |     zero_v2(tri_2d[1]); | 
					
						
							|  |  |  | #  endif
 | 
					
						
							|  |  |  |     mul_v2_m3v3_center(tri_2d[2], axis_mat, l_ear->next->v->co, center); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   /* check we're not flipping face corners before or after the ear */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     float adjacent_2d[2]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     if (!BM_vert_is_edge_pair(l_ear->prev->v)) { | 
					
						
							|  |  |  |       mul_v2_m3v3_center(adjacent_2d, axis_mat, l_ear->prev->prev->v->co, center); | 
					
						
							|  |  |  |       if (signum_i(cross_tri_v2(adjacent_2d, tri_2d[0], tri_2d[1])) != | 
					
						
							|  |  |  |           signum_i(cross_tri_v2(adjacent_2d, tri_2d[0], tri_2d[2]))) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     if (!BM_vert_is_edge_pair(l_ear->next->v)) { | 
					
						
							|  |  |  |       mul_v2_m3v3_center(adjacent_2d, axis_mat, l_ear->next->next->v->co, center); | 
					
						
							|  |  |  |       if (signum_i(cross_tri_v2(adjacent_2d, tri_2d[2], tri_2d[1])) != | 
					
						
							|  |  |  |           signum_i(cross_tri_v2(adjacent_2d, tri_2d[2], tri_2d[0]))) { | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   /* check no existing verts are inside the triangle */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     /* triangle may be concave, if so - flip so we can use clockwise check */ | 
					
						
							|  |  |  |     if (cross_tri_v2(UNPACK3(tri_2d)) < 0.0f) { | 
					
						
							|  |  |  |       swap_v2_v2(tri_2d[1], tri_2d[2]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     /* skip l_ear and adjacent verts */ | 
					
						
							|  |  |  |     BMLoop *l_iter, *l_first; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     l_iter = l_ear->next->next; | 
					
						
							|  |  |  |     l_first = l_ear->prev; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |       float co_2d[2]; | 
					
						
							|  |  |  |       mul_v2_m3v3_center(co_2d, axis_mat, l_iter->v->co, center); | 
					
						
							|  |  |  |       if (isect_point_tri_v2_cw(co_2d, tri_2d[0], tri_2d[1], tri_2d[2])) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } while ((l_iter = l_iter->next) != l_first); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool bm_vert_collapse_is_degenerate(BMVert *v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BMEdge *e_pair[2]; | 
					
						
							|  |  |  |   BMVert *v_pair[2]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |   if (BM_vert_edge_pair(v, &e_pair[0], &e_pair[1])) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-28 12:48:13 +10:00
										 |  |  |     /* allow wire edges */ | 
					
						
							|  |  |  |     if (BM_edge_is_wire(e_pair[0]) || BM_edge_is_wire(e_pair[1])) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     v_pair[0] = BM_edge_other_vert(e_pair[0], v); | 
					
						
							|  |  |  |     v_pair[1] = BM_edge_other_vert(e_pair[1], v); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |     if (fabsf(cos_v3v3v3(v_pair[0]->co, v->co, v_pair[1]->co)) < (1.0f - FLT_EPSILON)) { | 
					
						
							|  |  |  |       BMLoop *l_iter, *l_first; | 
					
						
							|  |  |  |       l_iter = l_first = e_pair[1]->l; | 
					
						
							|  |  |  |       do { | 
					
						
							|  |  |  |         if (l_iter->f->len > 3) { | 
					
						
							|  |  |  |           BMLoop *l_pivot = (l_iter->v == v ? l_iter : l_iter->next); | 
					
						
							|  |  |  |           BLI_assert(v == l_pivot->v); | 
					
						
							|  |  |  |           if (bm_loop_collapse_is_degenerate(l_pivot)) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } while ((l_iter = l_iter->radial_next) != l_first); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-07-03 14:37:53 +02:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  | } | 
					
						
							|  |  |  | #endif /* USE_DEGENERATE_CHECK */
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-25 20:15:20 +10:00
										 |  |  | void BM_mesh_decimate_dissolve_ex(BMesh *bm, | 
					
						
							|  |  |  |                                   const float angle_limit, | 
					
						
							|  |  |  |                                   const bool do_dissolve_boundaries, | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |                                   BMO_Delimit delimit, | 
					
						
							| 
									
										
										
										
											2015-04-25 20:15:20 +10:00
										 |  |  |                                   BMVert **vinput_arr, | 
					
						
							|  |  |  |                                   const int vinput_len, | 
					
						
							|  |  |  |                                   BMEdge **einput_arr, | 
					
						
							|  |  |  |                                   const int einput_len, | 
					
						
							|  |  |  |                                   const short oflag_out) | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-04-19 19:13:39 +10:00
										 |  |  |   const float angle_limit_cos_neg = -cosf(angle_limit); | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |   struct DelimitData delimit_data = {0}; | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   const int eheap_table_len = do_dissolve_boundaries ? einput_len : max_ii(einput_len, vinput_len); | 
					
						
							|  |  |  |   void *_heap_table = MEM_mallocN(sizeof(HeapNode *) * eheap_table_len, __func__); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   int i; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |   if (delimit & BMO_DELIM_UV) { | 
					
						
							|  |  |  |     const int layer_len = CustomData_number_of_layers(&bm->ldata, CD_MLOOPUV); | 
					
						
							|  |  |  |     if (layer_len == 0) { | 
					
						
							|  |  |  |       delimit &= ~BMO_DELIM_UV; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       delimit_data.cd_loop_type = CD_MLOOPUV; | 
					
						
							|  |  |  |       delimit_data.cd_loop_size = CustomData_sizeof(delimit_data.cd_loop_type); | 
					
						
							|  |  |  |       delimit_data.cd_loop_offset = CustomData_get_n_offset(&bm->ldata, CD_MLOOPUV, 0); | 
					
						
							|  |  |  |       delimit_data.cd_loop_offset_end = delimit_data.cd_loop_size * layer_len; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |   /* --- first edges --- */ | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   if (1) { | 
					
						
							|  |  |  |     BMEdge **earray; | 
					
						
							|  |  |  |     Heap *eheap; | 
					
						
							|  |  |  |     HeapNode **eheap_table = _heap_table; | 
					
						
							|  |  |  |     HeapNode *enode_top; | 
					
						
							|  |  |  |     int *vert_reverse_lookup; | 
					
						
							|  |  |  |     BMIter iter; | 
					
						
							|  |  |  |     BMEdge *e_iter; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     /* --- setup heap --- */ | 
					
						
							|  |  |  |     eheap = BLI_heap_new_ex(einput_len); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     /* wire -> tag */ | 
					
						
							|  |  |  |     BM_ITER_MESH (e_iter, &iter, bm, BM_EDGES_OF_MESH) { | 
					
						
							|  |  |  |       BM_elem_flag_set(e_iter, BM_ELEM_TAG, BM_edge_is_wire(e_iter)); | 
					
						
							|  |  |  |       BM_elem_index_set(e_iter, -1); /* set dirty */ | 
					
						
							| 
									
										
										
										
											2013-06-03 05:07:16 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     bm->elem_index_dirty |= BM_EDGE; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     /* build heap */ | 
					
						
							|  |  |  |     for (i = 0; i < einput_len; i++) { | 
					
						
							|  |  |  |       BMEdge *e = einput_arr[i]; | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |       const float cost = bm_edge_calc_dissolve_error(e, delimit, &delimit_data); | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       eheap_table[i] = BLI_heap_insert(eheap, cost, e); | 
					
						
							|  |  |  |       BM_elem_index_set(e, i); /* set dirty */ | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     while ((BLI_heap_is_empty(eheap) == false) && | 
					
						
							| 
									
										
										
										
											2016-04-19 19:13:39 +10:00
										 |  |  |            (BLI_heap_node_value((enode_top = BLI_heap_top(eheap))) < angle_limit_cos_neg)) { | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       BMFace *f_new = NULL; | 
					
						
							|  |  |  |       BMEdge *e; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       e = BLI_heap_node_ptr(enode_top); | 
					
						
							|  |  |  |       i = BM_elem_index_get(e); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       if (BM_edge_is_manifold(e)) { | 
					
						
							| 
									
										
										
										
											2016-11-12 10:06:53 +11:00
										 |  |  |         f_new = BM_faces_join_pair(bm, e->l, e->l->radial_next, false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |         if (f_new) { | 
					
						
							|  |  |  |           BMLoop *l_first, *l_iter; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           BLI_heap_remove(eheap, enode_top); | 
					
						
							|  |  |  |           eheap_table[i] = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           /* update normal */ | 
					
						
							| 
									
										
										
										
											2013-03-12 08:50:02 +00:00
										 |  |  |           BM_face_normal_update(f_new); | 
					
						
							| 
									
										
										
										
											2013-07-25 06:05:44 +00:00
										 |  |  |           if (oflag_out) { | 
					
						
							| 
									
										
										
										
											2016-07-01 19:07:11 +10:00
										 |  |  |             BMO_face_flag_enable(bm, f_new, oflag_out); | 
					
						
							| 
									
										
										
										
											2013-07-25 06:05:44 +00:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           /* re-calculate costs */ | 
					
						
							|  |  |  |           l_iter = l_first = BM_FACE_FIRST_LOOP(f_new); | 
					
						
							|  |  |  |           do { | 
					
						
							|  |  |  |             const int j = BM_elem_index_get(l_iter->e); | 
					
						
							|  |  |  |             if (j != -1 && eheap_table[j]) { | 
					
						
							| 
									
										
										
										
											2015-05-16 12:21:31 +10:00
										 |  |  |               const float cost = bm_edge_calc_dissolve_error(l_iter->e, delimit, &delimit_data); | 
					
						
							| 
									
										
										
										
											2017-10-30 00:36:52 +11:00
										 |  |  |               BLI_heap_node_value_update(eheap, eheap_table[j], cost); | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |           } while ((l_iter = l_iter->next) != l_first); | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           BMO_error_clear(bm); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       if (UNLIKELY(f_new == NULL)) { | 
					
						
							| 
									
										
										
										
											2017-10-30 00:36:52 +11:00
										 |  |  |         BLI_heap_node_value_update(eheap, enode_top, COST_INVALID); | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     /* prepare for cleanup */ | 
					
						
							|  |  |  |     BM_mesh_elem_index_ensure(bm, BM_VERT); | 
					
						
							|  |  |  |     vert_reverse_lookup = MEM_mallocN(sizeof(int) * bm->totvert, __func__); | 
					
						
							| 
									
										
										
										
											2015-05-05 17:08:29 +10:00
										 |  |  |     copy_vn_i(vert_reverse_lookup, bm->totvert, -1); | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     for (i = 0; i < vinput_len; i++) { | 
					
						
							|  |  |  |       BMVert *v = vinput_arr[i]; | 
					
						
							|  |  |  |       vert_reverse_lookup[BM_elem_index_get(v)] = i; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     /* --- cleanup --- */ | 
					
						
							|  |  |  |     earray = MEM_mallocN(sizeof(BMEdge *) * bm->totedge, __func__); | 
					
						
							|  |  |  |     BM_ITER_MESH_INDEX (e_iter, &iter, bm, BM_EDGES_OF_MESH, i) { | 
					
						
							|  |  |  |       earray[i] = e_iter; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-29 22:04:24 +10:00
										 |  |  |     /* Remove all edges/verts left behind from dissolving,
 | 
					
						
							| 
									
										
										
										
											2021-03-18 09:35:12 +11:00
										 |  |  |      * NULL'ing the vertex array so we don't re-use. */ | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     for (i = bm->totedge - 1; i != -1; i--) { | 
					
						
							|  |  |  |       e_iter = earray[i]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       if (BM_edge_is_wire(e_iter) && (BM_elem_flag_test(e_iter, BM_ELEM_TAG) == false)) { | 
					
						
							|  |  |  |         /* edge has become wire */ | 
					
						
							|  |  |  |         int vidx_reverse; | 
					
						
							|  |  |  |         BMVert *v1 = e_iter->v1; | 
					
						
							|  |  |  |         BMVert *v2 = e_iter->v2; | 
					
						
							|  |  |  |         BM_edge_kill(bm, e_iter); | 
					
						
							|  |  |  |         if (v1->e == NULL) { | 
					
						
							|  |  |  |           vidx_reverse = vert_reverse_lookup[BM_elem_index_get(v1)]; | 
					
						
							| 
									
										
										
										
											2019-03-27 17:14:36 +11:00
										 |  |  |           if (vidx_reverse != -1) { | 
					
						
							|  |  |  |             vinput_arr[vidx_reverse] = NULL; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           BM_vert_kill(bm, v1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (v2->e == NULL) { | 
					
						
							|  |  |  |           vidx_reverse = vert_reverse_lookup[BM_elem_index_get(v2)]; | 
					
						
							| 
									
										
										
										
											2019-03-27 17:14:36 +11:00
										 |  |  |           if (vidx_reverse != -1) { | 
					
						
							|  |  |  |             vinput_arr[vidx_reverse] = NULL; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           BM_vert_kill(bm, v2); | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     MEM_freeN(vert_reverse_lookup); | 
					
						
							|  |  |  |     MEM_freeN(earray); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     BLI_heap_free(eheap, NULL); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |   /* --- second verts --- */ | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |   if (do_dissolve_boundaries) { | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     /* simple version of the branch below, since we will dissolve _all_ verts that use 2 edges */ | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |     for (i = 0; i < vinput_len; i++) { | 
					
						
							|  |  |  |       BMVert *v = vinput_arr[i]; | 
					
						
							| 
									
										
										
										
											2014-06-27 20:22:19 +10:00
										 |  |  |       if (LIKELY(v != NULL) && BM_vert_is_edge_pair(v)) { | 
					
						
							| 
									
										
										
										
											2014-06-27 20:11:23 +10:00
										 |  |  |         BM_vert_collapse_edge(bm, v->e, v, true, true); /* join edges */ | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |   else { | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     Heap *vheap; | 
					
						
							|  |  |  |     HeapNode **vheap_table = _heap_table; | 
					
						
							|  |  |  |     HeapNode *vnode_top; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     BMVert *v_iter; | 
					
						
							|  |  |  |     BMIter iter; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     BM_ITER_MESH (v_iter, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  |       BM_elem_index_set(v_iter, -1); /* set dirty */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     bm->elem_index_dirty |= BM_VERT; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     vheap = BLI_heap_new_ex(vinput_len); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     for (i = 0; i < vinput_len; i++) { | 
					
						
							|  |  |  |       BMVert *v = vinput_arr[i]; | 
					
						
							|  |  |  |       if (LIKELY(v != NULL)) { | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |         const float cost = bm_vert_edge_face_angle(v, delimit, &delimit_data); | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |         vheap_table[i] = BLI_heap_insert(vheap, cost, v); | 
					
						
							|  |  |  |         BM_elem_index_set(v, i); /* set dirty */ | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     while ((BLI_heap_is_empty(vheap) == false) && | 
					
						
							|  |  |  |            (BLI_heap_node_value((vnode_top = BLI_heap_top(vheap))) < angle_limit)) { | 
					
						
							|  |  |  |       BMEdge *e_new = NULL; | 
					
						
							|  |  |  |       BMVert *v; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       v = BLI_heap_node_ptr(vnode_top); | 
					
						
							|  |  |  |       i = BM_elem_index_get(v); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |       if ( | 
					
						
							|  |  |  | #ifdef USE_DEGENERATE_CHECK
 | 
					
						
							|  |  |  |           !bm_vert_collapse_is_degenerate(v) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |           BM_vert_is_edge_pair(v) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       ) { | 
					
						
							| 
									
										
										
										
											2014-06-27 20:11:23 +10:00
										 |  |  |         e_new = BM_vert_collapse_edge(bm, v->e, v, true, true); /* join edges */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |         if (e_new) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           BLI_heap_remove(vheap, vnode_top); | 
					
						
							|  |  |  |           vheap_table[i] = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           /* update normal */ | 
					
						
							|  |  |  |           if (e_new->l) { | 
					
						
							|  |  |  |             BMLoop *l_first, *l_iter; | 
					
						
							|  |  |  |             l_iter = l_first = e_new->l; | 
					
						
							|  |  |  |             do { | 
					
						
							|  |  |  |               BM_face_normal_update(l_iter->f); | 
					
						
							|  |  |  |             } while ((l_iter = l_iter->radial_next) != l_first); | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |           /* re-calculate costs */ | 
					
						
							| 
									
										
										
										
											2013-08-17 08:21:40 +00:00
										 |  |  |           BM_ITER_ELEM (v_iter, &iter, e_new, BM_VERTS_OF_EDGE) { | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |             const int j = BM_elem_index_get(v_iter); | 
					
						
							|  |  |  |             if (j != -1 && vheap_table[j]) { | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |               const float cost = bm_vert_edge_face_angle(v_iter, delimit, &delimit_data); | 
					
						
							| 
									
										
										
										
											2017-10-30 00:36:52 +11:00
										 |  |  |               BLI_heap_node_value_update(vheap, vheap_table[j], cost); | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef USE_DEGENERATE_CHECK
 | 
					
						
							|  |  |  |           /* dissolving a vertex may mean vertices we previously weren't able to dissolve
 | 
					
						
							| 
									
										
										
										
											2016-04-19 13:07:01 +10:00
										 |  |  |            * can now be re-evaluated. */ | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |           if (e_new->l) { | 
					
						
							|  |  |  |             BMLoop *l_first, *l_iter; | 
					
						
							|  |  |  |             l_iter = l_first = e_new->l; | 
					
						
							|  |  |  |             do { | 
					
						
							| 
									
										
										
										
											2016-04-19 13:07:01 +10:00
										 |  |  |               /* skip vertices part of this edge, evaluated above */ | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |               BMLoop *l_cycle_first, *l_cycle_iter; | 
					
						
							| 
									
										
										
										
											2016-04-19 13:07:01 +10:00
										 |  |  |               l_cycle_iter = l_iter->next->next; | 
					
						
							|  |  |  |               l_cycle_first = l_iter->prev; | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |               do { | 
					
						
							|  |  |  |                 const int j = BM_elem_index_get(l_cycle_iter->v); | 
					
						
							|  |  |  |                 if (j != -1 && vheap_table[j] && | 
					
						
							|  |  |  |                     (BLI_heap_node_value(vheap_table[j]) == COST_INVALID)) { | 
					
						
							| 
									
										
										
										
											2020-02-15 17:45:53 +11:00
										 |  |  |                   const float cost = bm_vert_edge_face_angle( | 
					
						
							|  |  |  |                       l_cycle_iter->v, delimit, &delimit_data); | 
					
						
							| 
									
										
										
										
											2017-10-30 00:36:52 +11:00
										 |  |  |                   BLI_heap_node_value_update(vheap, vheap_table[j], cost); | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |                 } | 
					
						
							|  |  |  |               } while ((l_cycle_iter = l_cycle_iter->next) != l_cycle_first); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:11:51 +10:00
										 |  |  |             } while ((l_iter = l_iter->radial_next) != l_first); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | #endif /* USE_DEGENERATE_CHECK */
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       if (UNLIKELY(e_new == NULL)) { | 
					
						
							| 
									
										
										
										
											2017-10-30 00:36:52 +11:00
										 |  |  |         BLI_heap_node_value_update(vheap, vnode_top, COST_INVALID); | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |     BLI_heap_free(vheap, NULL); | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 21:01:42 +00:00
										 |  |  |   MEM_freeN(_heap_table); | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-25 20:15:20 +10:00
										 |  |  | void BM_mesh_decimate_dissolve(BMesh *bm, | 
					
						
							|  |  |  |                                const float angle_limit, | 
					
						
							|  |  |  |                                const bool do_dissolve_boundaries, | 
					
						
							|  |  |  |                                const BMO_Delimit delimit) | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int vinput_len; | 
					
						
							|  |  |  |   int einput_len; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-12 05:53:43 +00:00
										 |  |  |   BMVert **vinput_arr = BM_iter_as_arrayN(bm, BM_VERTS_OF_MESH, NULL, &vinput_len, NULL, 0); | 
					
						
							|  |  |  |   BMEdge **einput_arr = BM_iter_as_arrayN(bm, BM_EDGES_OF_MESH, NULL, &einput_len, NULL, 0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 06:13:56 +00:00
										 |  |  |   BM_mesh_decimate_dissolve_ex(bm, | 
					
						
							|  |  |  |                                angle_limit, | 
					
						
							|  |  |  |                                do_dissolve_boundaries, | 
					
						
							| 
									
										
										
										
											2013-06-03 05:07:16 +00:00
										 |  |  |                                delimit, | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |                                vinput_arr, | 
					
						
							|  |  |  |                                vinput_len, | 
					
						
							| 
									
										
										
										
											2013-07-25 06:05:44 +00:00
										 |  |  |                                einput_arr, | 
					
						
							|  |  |  |                                einput_len, | 
					
						
							|  |  |  |                                0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 05:20:02 +00:00
										 |  |  |   MEM_freeN(vinput_arr); | 
					
						
							|  |  |  |   MEM_freeN(einput_arr); | 
					
						
							|  |  |  | } |