While \file doesn't need an argument, it can't have another doxy command after it.
		
			
				
	
	
		
			206 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			206 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| /** \file
 | |
|  * \ingroup bke
 | |
|  *
 | |
|  * Functions for iterating mesh features.
 | |
|  */
 | |
| 
 | |
| #include "DNA_mesh_types.h"
 | |
| #include "DNA_meshdata_types.h"
 | |
| 
 | |
| #include "BKE_customdata.h"
 | |
| #include "BKE_mesh.h"
 | |
| #include "BKE_mesh_iterators.h"
 | |
| 
 | |
| #include "BLI_bitmap.h"
 | |
| #include "BLI_math.h"
 | |
| 
 | |
| #include "MEM_guardedalloc.h"
 | |
| 
 | |
| /* Copied from cdDM_foreachMappedVert */
 | |
| void BKE_mesh_foreach_mapped_vert(
 | |
|         Mesh *mesh,
 | |
|         void (*func)(void *userData, int index, const float co[3], const float no_f[3], const short no_s[3]),
 | |
|         void *userData,
 | |
|         MeshForeachFlag flag)
 | |
| {
 | |
| 	const MVert *mv = mesh->mvert;
 | |
| 	const int *index = CustomData_get_layer(&mesh->vdata, CD_ORIGINDEX);
 | |
| 
 | |
| 	if (index) {
 | |
| 		for (int i = 0; i < mesh->totvert; i++, mv++) {
 | |
| 			const short *no = (flag & MESH_FOREACH_USE_NORMAL) ? mv->no : NULL;
 | |
| 			const int orig = *index++;
 | |
| 			if (orig == ORIGINDEX_NONE) {
 | |
| 				continue;
 | |
| 			}
 | |
| 			func(userData, orig, mv->co, NULL, no);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		for (int i = 0; i < mesh->totvert; i++, mv++) {
 | |
| 			const short *no = (flag & MESH_FOREACH_USE_NORMAL) ? mv->no : NULL;
 | |
| 			func(userData, i, mv->co, NULL, no);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* Copied from cdDM_foreachMappedEdge */
 | |
| void BKE_mesh_foreach_mapped_edge(
 | |
|         Mesh *mesh,
 | |
|         void (*func)(void *userData, int index, const float v0co[3], const float v1co[3]),
 | |
|         void *userData)
 | |
| {
 | |
| 	const MVert *mv = mesh->mvert;
 | |
| 	const MEdge *med = mesh->medge;
 | |
| 	const int *index = CustomData_get_layer(&mesh->edata, CD_ORIGINDEX);
 | |
| 
 | |
| 	if (index) {
 | |
| 		for (int i = 0; i < mesh->totedge; i++, med++) {
 | |
| 			const int orig = *index++;
 | |
| 			if (orig == ORIGINDEX_NONE) {
 | |
| 				continue;
 | |
| 			}
 | |
| 			func(userData, orig, mv[med->v1].co, mv[med->v2].co);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		for (int i = 0; i < mesh->totedge; i++, med++) {
 | |
| 			func(userData, i, mv[med->v1].co, mv[med->v2].co);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* Copied from cdDM_foreachMappedLoop */
 | |
| void BKE_mesh_foreach_mapped_loop(
 | |
|         Mesh *mesh,
 | |
|         void (*func)(void *userData, int vertex_index, int face_index, const float co[3], const float no[3]),
 | |
|         void *userData,
 | |
|         MeshForeachFlag flag)
 | |
| {
 | |
| 	/* We can't use dm->getLoopDataLayout(dm) here, we want to always access dm->loopData, EditDerivedBMesh would
 | |
| 	 * return loop data from bmesh itself. */
 | |
| 	const float (*lnors)[3] = (flag & MESH_FOREACH_USE_NORMAL) ? CustomData_get_layer(&mesh->ldata, CD_NORMAL) : NULL;
 | |
| 
 | |
| 	const MVert *mv = mesh->mvert;
 | |
| 	const MLoop *ml = mesh->mloop;
 | |
| 	const MPoly *mp = mesh->mpoly;
 | |
| 	const int *v_index = CustomData_get_layer(&mesh->vdata, CD_ORIGINDEX);
 | |
| 	const int *f_index = CustomData_get_layer(&mesh->pdata, CD_ORIGINDEX);
 | |
| 	int p_idx, i;
 | |
| 
 | |
| 	if (v_index || f_index) {
 | |
| 		for (p_idx = 0; p_idx < mesh->totpoly; p_idx++, mp++) {
 | |
| 			for (i = 0; i < mp->totloop; i++, ml++) {
 | |
| 				const int v_idx = v_index ? v_index[ml->v] : ml->v;
 | |
| 				const int f_idx = f_index ? f_index[p_idx] : p_idx;
 | |
| 				const float *no = lnors ? *lnors++ : NULL;
 | |
| 				if (ELEM(ORIGINDEX_NONE, v_idx, f_idx)) {
 | |
| 					continue;
 | |
| 				}
 | |
| 				func(userData, v_idx, f_idx, mv[ml->v].co, no);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		for (p_idx = 0; p_idx < mesh->totpoly; p_idx++, mp++) {
 | |
| 			for (i = 0; i < mp->totloop; i++, ml++) {
 | |
| 				const int v_idx = ml->v;
 | |
| 				const int f_idx = p_idx;
 | |
| 				const float *no = lnors ? *lnors++ : NULL;
 | |
| 				func(userData, v_idx, f_idx, mv[ml->v].co, no);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* Copied from cdDM_foreachMappedFaceCenter */
 | |
| void BKE_mesh_foreach_mapped_face_center(
 | |
|         Mesh *mesh,
 | |
|         void (*func)(void *userData, int index, const float cent[3], const float no[3]),
 | |
|         void *userData,
 | |
|         MeshForeachFlag flag)
 | |
| {
 | |
| 	const MVert *mvert = mesh->mvert;
 | |
| 	const MPoly *mp = mesh->mpoly;
 | |
| 	const MLoop *ml;
 | |
| 	float _no_buf[3];
 | |
| 	float *no = (flag & MESH_FOREACH_USE_NORMAL) ? _no_buf : NULL;
 | |
| 	const int *index = CustomData_get_layer(&mesh->pdata, CD_ORIGINDEX);
 | |
| 
 | |
| 	if (index) {
 | |
| 		for (int i = 0; i < mesh->totpoly; i++, mp++) {
 | |
| 			const int orig = *index++;
 | |
| 			if (orig == ORIGINDEX_NONE) {
 | |
| 				continue;
 | |
| 			}
 | |
| 			float cent[3];
 | |
| 			ml = &mesh->mloop[mp->loopstart];
 | |
| 			BKE_mesh_calc_poly_center(mp, ml, mvert, cent);
 | |
| 			if (flag & MESH_FOREACH_USE_NORMAL) {
 | |
| 				BKE_mesh_calc_poly_normal(mp, ml, mvert, no);
 | |
| 			}
 | |
| 			func(userData, orig, cent, no);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		for (int i = 0; i < mesh->totpoly; i++, mp++) {
 | |
| 			float cent[3];
 | |
| 			ml = &mesh->mloop[mp->loopstart];
 | |
| 			BKE_mesh_calc_poly_center(mp, ml, mvert, cent);
 | |
| 			if (flag & MESH_FOREACH_USE_NORMAL) {
 | |
| 				BKE_mesh_calc_poly_normal(mp, ml, mvert, no);
 | |
| 			}
 | |
| 			func(userData, i, cent, no);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Helpers based on above foreach loopers> */
 | |
| 
 | |
| typedef struct MappedVCosData {
 | |
| 	float (*vertexcos)[3];
 | |
| 	BLI_bitmap *vertex_visit;
 | |
| } MappedVCosData;
 | |
| 
 | |
| static void get_vertexcos__mapFunc(
 | |
|         void *user_data, int index, const float co[3],
 | |
|         const float UNUSED(no_f[3]), const short UNUSED(no_s[3]))
 | |
| {
 | |
| 	MappedVCosData *mapped_vcos_data = (MappedVCosData *)user_data;
 | |
| 
 | |
| 	if (BLI_BITMAP_TEST(mapped_vcos_data->vertex_visit, index) == 0) {
 | |
| 		/* We need coord from prototype vertex, not from copies,
 | |
| 		 * we assume they stored in the beginning of vertex array stored in evaluated mesh
 | |
| 		 * (mirror modifier for eg does this). */
 | |
| 		copy_v3_v3(mapped_vcos_data->vertexcos[index], co);
 | |
| 		BLI_BITMAP_ENABLE(mapped_vcos_data->vertex_visit, index);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void BKE_mesh_foreach_mapped_vert_coords_get(Mesh *me_eval, float (*r_cos)[3], const int totcos)
 | |
| {
 | |
| 	MappedVCosData user_data;
 | |
| 	memset(r_cos, 0, sizeof(*r_cos) * totcos);
 | |
| 	user_data.vertexcos = r_cos;
 | |
| 	user_data.vertex_visit = BLI_BITMAP_NEW(totcos, __func__);
 | |
| 	BKE_mesh_foreach_mapped_vert(me_eval, get_vertexcos__mapFunc, &user_data, MESH_FOREACH_NOP);
 | |
| 	MEM_freeN(user_data.vertex_visit);
 | |
| }
 |