| 
									
										
										
										
											2013-04-16 05:46:17 +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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2005 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bke | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | #include "DNA_listBase.h"
 | 
					
						
							|  |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 14:24:57 +11:00
										 |  |  | #include "BLI_bitmap.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-26 14:24:57 +11:00
										 |  |  | #include "BKE_DerivedMesh.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | #include "BKE_editmesh.h"
 | 
					
						
							| 
									
										
										
										
											2020-05-25 20:16:42 +10:00
										 |  |  | #include "BKE_editmesh_cache.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-10 12:58:59 +01:00
										 |  |  | #include "BKE_lib_id.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:36:45 +02:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  | #include "BKE_mesh_iterators.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-10 22:32:06 +10:00
										 |  |  | #include "BKE_mesh_wrapper.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-23 14:36:45 +02:00
										 |  |  | #include "BKE_object.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | BMEditMesh *BKE_editmesh_create(BMesh *bm, const bool do_tessellate) | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BMEditMesh *em = MEM_callocN(sizeof(BMEditMesh), __func__); | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   em->bm = bm; | 
					
						
							|  |  |  |   if (do_tessellate) { | 
					
						
							| 
									
										
										
										
											2019-08-25 14:32:47 +10:00
										 |  |  |     BKE_editmesh_looptri_calc(em); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return em; | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | BMEditMesh *BKE_editmesh_copy(BMEditMesh *em) | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BMEditMesh *em_copy = MEM_callocN(sizeof(BMEditMesh), __func__); | 
					
						
							|  |  |  |   *em_copy = *em; | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   em_copy->mesh_eval_cage = em_copy->mesh_eval_final = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-23 14:36:45 +02:00
										 |  |  |   em_copy->bb_cage = NULL; | 
					
						
							| 
									
										
										
										
											2013-04-27 23:42:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   em_copy->bm = BM_mesh_copy(em->bm); | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* The tessellation is NOT calculated on the copy here,
 | 
					
						
							|  |  |  |    * because currently all the callers of this function use | 
					
						
							| 
									
										
										
										
											2019-08-17 00:54:22 +10:00
										 |  |  |    * it to make a backup copy of the #BMEditMesh to restore | 
					
						
							|  |  |  |    * it in the case of errors in an operation. For performance reasons, | 
					
						
							|  |  |  |    * in that case it makes more sense to do the | 
					
						
							|  |  |  |    * tessellation only when/if that copy ends up getting used. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   em_copy->looptris = NULL; | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 16:24:33 +10:00
										 |  |  |   /* Copy various settings. */ | 
					
						
							|  |  |  |   em_copy->selectmode = em->selectmode; | 
					
						
							|  |  |  |   em_copy->mat_nr = em->mat_nr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return em_copy; | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * \brief Return the BMEditMesh for a given object | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note this function assumes this is a mesh object, | 
					
						
							|  |  |  |  * don't add NULL data check here. caller must do that | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | BMEditMesh *BKE_editmesh_from_object(Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(ob->type == OB_MESH); | 
					
						
							|  |  |  |   /* sanity check */ | 
					
						
							| 
									
										
										
										
											2018-04-16 16:27:55 +02:00
										 |  |  | #if 0 /* disable in mutlti-object edit. */
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #  ifndef NDEBUG
 | 
					
						
							|  |  |  |   if (((Mesh *)ob->data)->edit_mesh) { | 
					
						
							|  |  |  |     BLI_assert(((Mesh *)ob->data)->edit_mesh->ob == ob); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #  endif
 | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return ((Mesh *)ob->data)->edit_mesh; | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 22:56:03 +10:00
										 |  |  | static void editmesh_tessface_calc_intern(BMEditMesh *em, | 
					
						
							|  |  |  |                                           const struct BMeshCalcTessellation_Params *params) | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* allocating space before calculating the tessellation */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BMesh *bm = em->bm; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-17 00:54:22 +10:00
										 |  |  |   /* This assumes all faces can be scan-filled, which isn't always true,
 | 
					
						
							|  |  |  |    * worst case we over allocate a little which is acceptable. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int looptris_tot = poly_to_tri_count(bm->totface, bm->totloop); | 
					
						
							|  |  |  |   const int looptris_tot_prev_alloc = em->looptris ? | 
					
						
							|  |  |  |                                           (MEM_allocN_len(em->looptris) / sizeof(*em->looptris)) : | 
					
						
							|  |  |  |                                           0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BMLoop *(*looptris)[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* this means no reallocs for quad dominant models, for */ | 
					
						
							|  |  |  |   if ((em->looptris != NULL) && | 
					
						
							|  |  |  |       /* (*em->tottri >= looptris_tot)) */ | 
					
						
							| 
									
										
										
										
											2019-08-17 00:54:22 +10:00
										 |  |  |       /* Check against allocated size in case we over allocated a little. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       ((looptris_tot_prev_alloc >= looptris_tot) && | 
					
						
							|  |  |  |        (looptris_tot_prev_alloc <= looptris_tot * 2))) { | 
					
						
							|  |  |  |     looptris = em->looptris; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2019-04-22 09:39:35 +10:00
										 |  |  |     if (em->looptris) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       MEM_freeN(em->looptris); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:39:35 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     looptris = MEM_mallocN(sizeof(*looptris) * looptris_tot, __func__); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   em->looptris = looptris; | 
					
						
							| 
									
										
										
										
											2021-06-01 12:49:22 +10:00
										 |  |  |   em->tottri = looptris_tot; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* after allocating the em->looptris, we're ready to tessellate */ | 
					
						
							| 
									
										
										
										
											2021-06-14 22:56:03 +10:00
										 |  |  |   BM_mesh_calc_tessellation_ex(em->bm, em->looptris, params); | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 22:56:03 +10:00
										 |  |  | void BKE_editmesh_looptri_calc_ex(BMEditMesh *em, | 
					
						
							|  |  |  |                                   const struct BMeshCalcTessellation_Params *params) | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-14 22:56:03 +10:00
										 |  |  |   editmesh_tessface_calc_intern(em, params); | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* commented because editbmesh_build_data() ensures we get tessfaces */ | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (em->mesh_eval_final && em->mesh_eval_final == em->mesh_eval_cage) { | 
					
						
							|  |  |  |     BKE_mesh_runtime_looptri_ensure(em->mesh_eval_final); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (em->mesh_eval_final) { | 
					
						
							|  |  |  |     BKE_mesh_runtime_looptri_ensure(em->mesh_eval_final); | 
					
						
							|  |  |  |     BKE_mesh_runtime_looptri_ensure(em->mesh_eval_cage); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 22:56:03 +10:00
										 |  |  | void BKE_editmesh_looptri_calc(BMEditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BKE_editmesh_looptri_calc_ex(em, | 
					
						
							|  |  |  |                                &(const struct BMeshCalcTessellation_Params){ | 
					
						
							|  |  |  |                                    .face_normals = false, | 
					
						
							|  |  |  |                                }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Performing the face normal calculation at the same time as tessellation | 
					
						
							|  |  |  |  * gives a reasonable performance boost (approx ~20% faster). | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void BKE_editmesh_looptri_and_normals_calc(BMEditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BKE_editmesh_looptri_calc_ex(em, | 
					
						
							|  |  |  |                                &(const struct BMeshCalcTessellation_Params){ | 
					
						
							|  |  |  |                                    .face_normals = true, | 
					
						
							|  |  |  |                                }); | 
					
						
							|  |  |  |   BM_mesh_normals_update_ex(em->bm, | 
					
						
							|  |  |  |                             &(const struct BMeshNormalsUpdate_Params){ | 
					
						
							|  |  |  |                                 .face_normals = false, | 
					
						
							|  |  |  |                             }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_editmesh_looptri_calc_with_partial_ex(BMEditMesh *em, | 
					
						
							|  |  |  |                                                struct BMPartialUpdate *bmpinfo, | 
					
						
							|  |  |  |                                                const struct BMeshCalcTessellation_Params *params) | 
					
						
							| 
									
										
										
										
											2021-05-28 18:18:13 +10:00
										 |  |  | { | 
					
						
							|  |  |  |   BLI_assert(em->tottri == poly_to_tri_count(em->bm->totface, em->bm->totloop)); | 
					
						
							|  |  |  |   BLI_assert(em->looptris != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 22:56:03 +10:00
										 |  |  |   BM_mesh_calc_tessellation_with_partial_ex(em->bm, em->looptris, bmpinfo, params); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_editmesh_looptri_calc_with_partial(BMEditMesh *em, struct BMPartialUpdate *bmpinfo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BKE_editmesh_looptri_calc_with_partial_ex(em, bmpinfo, false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_editmesh_looptri_and_normals_calc_with_partial(BMEditMesh *em, | 
					
						
							|  |  |  |                                                         struct BMPartialUpdate *bmpinfo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BKE_editmesh_looptri_calc_with_partial_ex(em, | 
					
						
							|  |  |  |                                             bmpinfo, | 
					
						
							|  |  |  |                                             &(const struct BMeshCalcTessellation_Params){ | 
					
						
							|  |  |  |                                                 .face_normals = true, | 
					
						
							|  |  |  |                                             }); | 
					
						
							|  |  |  |   BM_mesh_normals_update_with_partial_ex(em->bm, | 
					
						
							|  |  |  |                                          bmpinfo, | 
					
						
							|  |  |  |                                          &(const struct BMeshNormalsUpdate_Params){ | 
					
						
							|  |  |  |                                              .face_normals = false, | 
					
						
							|  |  |  |                                          }); | 
					
						
							| 
									
										
										
										
											2021-05-28 18:18:13 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-28 20:02:30 +11:00
										 |  |  | void BKE_editmesh_free_derivedmesh(BMEditMesh *em) | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (em->mesh_eval_cage) { | 
					
						
							|  |  |  |     BKE_id_free(NULL, em->mesh_eval_cage); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (em->mesh_eval_final && em->mesh_eval_final != em->mesh_eval_cage) { | 
					
						
							|  |  |  |     BKE_id_free(NULL, em->mesh_eval_final); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   em->mesh_eval_cage = em->mesh_eval_final = NULL; | 
					
						
							| 
									
										
										
										
											2019-09-23 14:36:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   MEM_SAFE_FREE(em->bb_cage); | 
					
						
							| 
									
										
										
										
											2014-02-28 20:02:30 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*does not free the BMEditMesh struct itself*/ | 
					
						
							|  |  |  | void BKE_editmesh_free(BMEditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BKE_editmesh_free_derivedmesh(em); | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:39:35 +10:00
										 |  |  |   if (em->looptris) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(em->looptris); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:39:35 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:39:35 +10:00
										 |  |  |   if (em->bm) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     BM_mesh_free(em->bm); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:39:35 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-04-16 05:46:17 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-04-18 17:09:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  | struct CageUserData { | 
					
						
							|  |  |  |   int totvert; | 
					
						
							|  |  |  |   float (*cos_cage)[3]; | 
					
						
							|  |  |  |   BLI_bitmap *visit_bitmap; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void cage_mapped_verts_callback(void *userData, | 
					
						
							|  |  |  |                                        int index, | 
					
						
							|  |  |  |                                        const float co[3], | 
					
						
							|  |  |  |                                        const float UNUSED(no_f[3]), | 
					
						
							|  |  |  |                                        const short UNUSED(no_s[3])) | 
					
						
							| 
									
										
										
										
											2013-04-18 17:09:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  |   struct CageUserData *data = userData; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  |   if ((index >= 0 && index < data->totvert) && (!BLI_BITMAP_TEST(data->visit_bitmap, index))) { | 
					
						
							|  |  |  |     BLI_BITMAP_ENABLE(data->visit_bitmap, index); | 
					
						
							|  |  |  |     copy_v3_v3(data->cos_cage[index], co); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-04-18 17:09:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-07 13:49:10 +11:00
										 |  |  | float (*BKE_editmesh_vert_coords_alloc(struct Depsgraph *depsgraph, | 
					
						
							|  |  |  |                                        BMEditMesh *em, | 
					
						
							|  |  |  |                                        struct Scene *scene, | 
					
						
							|  |  |  |                                        Object *ob, | 
					
						
							|  |  |  |                                        int *r_vert_len))[3] | 
					
						
							| 
									
										
										
										
											2013-04-18 17:09:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  |   Mesh *cage; | 
					
						
							|  |  |  |   BLI_bitmap *visit_bitmap; | 
					
						
							|  |  |  |   struct CageUserData data; | 
					
						
							|  |  |  |   float(*cos_cage)[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-07 13:49:10 +11:00
										 |  |  |   cage = editbmesh_get_eval_cage(depsgraph, scene, ob, em, &CD_MASK_BAREMESH); | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  |   cos_cage = MEM_callocN(sizeof(*cos_cage) * em->bm->totvert, "bmbvh cos_cage"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* when initializing cage verts, we only want the first cage coordinate for each vertex,
 | 
					
						
							|  |  |  |    * so that e.g. mirror or array use original vertex coordinates and not mirrored or duplicate */ | 
					
						
							|  |  |  |   visit_bitmap = BLI_BITMAP_NEW(em->bm->totvert, __func__); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   data.totvert = em->bm->totvert; | 
					
						
							|  |  |  |   data.cos_cage = cos_cage; | 
					
						
							|  |  |  |   data.visit_bitmap = visit_bitmap; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BKE_mesh_foreach_mapped_vert(cage, cage_mapped_verts_callback, &data, MESH_FOREACH_NOP); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_freeN(visit_bitmap); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (r_vert_len) { | 
					
						
							|  |  |  |     *r_vert_len = em->bm->totvert; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-12-24 22:32:43 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return cos_cage; | 
					
						
							| 
									
										
										
										
											2013-04-18 17:09:56 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-03-29 04:44:05 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-14 14:49:22 +11:00
										 |  |  | const float (*BKE_editmesh_vert_coords_when_deformed(struct Depsgraph *depsgraph, | 
					
						
							|  |  |  |                                                      BMEditMesh *em, | 
					
						
							|  |  |  |                                                      struct Scene *scene, | 
					
						
							|  |  |  |                                                      Object *ob, | 
					
						
							|  |  |  |                                                      int *r_vert_len, | 
					
						
							|  |  |  |                                                      bool *r_is_alloc))[3] | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const float(*coords)[3] = NULL; | 
					
						
							|  |  |  |   *r_is_alloc = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Mesh *me = ob->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ((me->runtime.edit_data != NULL) && (me->runtime.edit_data->vertexCos != NULL)) { | 
					
						
							|  |  |  |     /* Deformed, and we have deformed coords already. */ | 
					
						
							|  |  |  |     coords = me->runtime.edit_data->vertexCos; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if ((em->mesh_eval_final != NULL) && | 
					
						
							|  |  |  |            (em->mesh_eval_final->runtime.wrapper_type == ME_WRAPPER_TYPE_BMESH)) { | 
					
						
							|  |  |  |     /* If this is an edit-mesh type, leave NULL as we can use the vertex coords. . */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Constructive modifiers have been used, we need to allocate coordinates. */ | 
					
						
							|  |  |  |     *r_is_alloc = true; | 
					
						
							|  |  |  |     coords = BKE_editmesh_vert_coords_alloc(depsgraph, em, scene, ob, r_vert_len); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return coords; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-22 06:28:35 +10:00
										 |  |  | float (*BKE_editmesh_vert_coords_alloc_orco(BMEditMesh *em, int *r_vert_len))[3] | 
					
						
							| 
									
										
										
										
											2015-03-29 04:44:05 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-11-21 21:03:03 +11:00
										 |  |  |   return BM_mesh_vert_coords_alloc(em->bm, r_vert_len); | 
					
						
							| 
									
										
										
										
											2015-03-29 04:44:05 +11:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-05-25 22:24:24 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-07 14:20:49 +11:00
										 |  |  | void BKE_editmesh_lnorspace_update(BMEditMesh *em, Mesh *me) | 
					
						
							| 
									
										
										
										
											2018-05-25 22:24:24 +05:30
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BMesh *bm = em->bm; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-01 13:12:24 +10:00
										 |  |  |   /* We need to create custom-loop-normals (CLNORS) data if none exist yet,
 | 
					
						
							|  |  |  |    * otherwise there is no way to edit them. | 
					
						
							|  |  |  |    * Similar code to #MESH_OT_customdata_custom_splitnormals_add operator, | 
					
						
							|  |  |  |    * we want to keep same shading in case we were using auto-smooth so far. | 
					
						
							| 
									
										
										
										
											2019-04-27 12:07:07 +10:00
										 |  |  |    * Note: there is a problem here, which is that if someone starts a normal editing operation on | 
					
						
							| 
									
										
										
										
											2020-07-01 13:12:24 +10:00
										 |  |  |    * previously auto-smooth-ed mesh, and cancel that operation, generated CLNORS data remain, | 
					
						
							|  |  |  |    * with related sharp edges (and hence auto-smooth is 'lost'). | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |    * Not sure how critical this is, and how to fix that issue? */ | 
					
						
							|  |  |  |   if (!CustomData_has_layer(&bm->ldata, CD_CUSTOMLOOPNORMAL)) { | 
					
						
							|  |  |  |     if (me->flag & ME_AUTOSMOOTH) { | 
					
						
							|  |  |  |       BM_edges_sharp_from_angle_set(bm, me->smoothresh); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BM_lnorspace_update(bm); | 
					
						
							| 
									
										
										
										
											2018-05-25 22:24:24 +05:30
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-20 12:15:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* If autosmooth not already set, set it */ | 
					
						
							| 
									
										
										
										
											2020-01-07 14:20:49 +11:00
										 |  |  | void BKE_editmesh_ensure_autosmooth(BMEditMesh *em, Mesh *me) | 
					
						
							| 
									
										
										
										
											2019-05-20 12:15:52 -04:00
										 |  |  | { | 
					
						
							|  |  |  |   if (!(me->flag & ME_AUTOSMOOTH)) { | 
					
						
							|  |  |  |     me->flag |= ME_AUTOSMOOTH; | 
					
						
							| 
									
										
										
										
											2020-01-07 14:20:49 +11:00
										 |  |  |     BKE_editmesh_lnorspace_update(em, me); | 
					
						
							| 
									
										
										
										
											2019-05-20 12:15:52 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-23 14:36:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | BoundBox *BKE_editmesh_cage_boundbox_get(BMEditMesh *em) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (em->bb_cage == NULL) { | 
					
						
							|  |  |  |     float min[3], max[3]; | 
					
						
							|  |  |  |     INIT_MINMAX(min, max); | 
					
						
							|  |  |  |     if (em->mesh_eval_cage) { | 
					
						
							| 
									
										
										
										
											2020-05-25 20:16:42 +10:00
										 |  |  |       BKE_mesh_wrapper_minmax(em->mesh_eval_cage, min, max); | 
					
						
							| 
									
										
										
										
											2019-09-23 14:36:45 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     em->bb_cage = MEM_callocN(sizeof(BoundBox), "BMEditMesh.bb_cage"); | 
					
						
							|  |  |  |     BKE_boundbox_init_from_minmax(em->bb_cage, min, max); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return em->bb_cage; | 
					
						
							|  |  |  | } |