2011-10-10 09:38:02 +00:00
|
|
|
/*
|
2008-01-07 19:13:47 +00:00
|
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
2002-10-12 11:37:38 +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
|
2008-01-07 19:13:47 +00:00
|
|
|
* of the License, or (at your option) any later version.
|
2002-10-12 11:37:38 +00:00
|
|
|
*
|
|
|
|
* 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,
|
2010-02-12 13:34:04 +00:00
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2002-10-12 11:37:38 +00:00
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
2009-01-04 14:14:06 +00:00
|
|
|
* Contributor(s): Blender Foundation
|
2002-10-12 11:37:38 +00:00
|
|
|
*
|
2008-01-07 19:13:47 +00:00
|
|
|
* ***** END GPL LICENSE BLOCK *****
|
2002-10-12 11:37:38 +00:00
|
|
|
*/
|
|
|
|
|
2011-02-27 20:40:57 +00:00
|
|
|
/** \file blender/blenkernel/intern/mesh.c
|
|
|
|
* \ingroup bke
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
2009-10-22 23:22:05 +00:00
|
|
|
#include "DNA_scene_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "DNA_material_types.h"
|
|
|
|
#include "DNA_object_types.h"
|
|
|
|
#include "DNA_key_types.h"
|
2012-02-19 22:17:30 +00:00
|
|
|
#include "DNA_mesh_types.h"
|
2004-03-20 22:55:42 +00:00
|
|
|
#include "DNA_meshdata_types.h"
|
2006-09-12 14:05:26 +00:00
|
|
|
#include "DNA_ipo_types.h"
|
2010-10-27 02:22:55 +00:00
|
|
|
#include "DNA_customdata_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2011-05-09 04:06:48 +00:00
|
|
|
#include "BLI_utildefines.h"
|
2011-01-07 18:36:47 +00:00
|
|
|
#include "BLI_blenlib.h"
|
|
|
|
#include "BLI_math.h"
|
|
|
|
#include "BLI_edgehash.h"
|
2011-02-27 06:19:40 +00:00
|
|
|
#include "BLI_scanfill.h"
|
2012-05-21 06:33:45 +00:00
|
|
|
#include "BLI_array.h"
|
2011-01-07 18:36:47 +00:00
|
|
|
|
2009-12-28 03:45:24 +00:00
|
|
|
#include "BKE_animsys.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_main.h"
|
2010-10-27 02:22:55 +00:00
|
|
|
#include "BKE_customdata.h"
|
2005-03-29 16:43:39 +00:00
|
|
|
#include "BKE_DerivedMesh.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_global.h"
|
|
|
|
#include "BKE_mesh.h"
|
|
|
|
#include "BKE_displist.h"
|
|
|
|
#include "BKE_library.h"
|
|
|
|
#include "BKE_material.h"
|
2010-09-09 00:14:51 +00:00
|
|
|
#include "BKE_modifier.h"
|
|
|
|
#include "BKE_multires.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_key.h"
|
|
|
|
/* these 2 are only used by conversion functions */
|
|
|
|
#include "BKE_curve.h"
|
|
|
|
/* -- */
|
|
|
|
#include "BKE_object.h"
|
2013-04-13 20:31:52 +00:00
|
|
|
#include "BKE_editmesh.h"
|
2010-10-27 02:22:55 +00:00
|
|
|
#include "BLI_edgehash.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2009-05-26 04:17:47 +00:00
|
|
|
#include "bmesh.h"
|
|
|
|
|
2010-10-27 02:22:55 +00:00
|
|
|
enum {
|
|
|
|
MESHCMP_DVERT_WEIGHTMISMATCH = 1,
|
|
|
|
MESHCMP_DVERT_GROUPMISMATCH,
|
|
|
|
MESHCMP_DVERT_TOTGROUPMISMATCH,
|
|
|
|
MESHCMP_LOOPCOLMISMATCH,
|
|
|
|
MESHCMP_LOOPUVMISMATCH,
|
|
|
|
MESHCMP_LOOPMISMATCH,
|
|
|
|
MESHCMP_POLYVERTMISMATCH,
|
|
|
|
MESHCMP_POLYMISMATCH,
|
|
|
|
MESHCMP_EDGEUNKNOWN,
|
|
|
|
MESHCMP_VERTCOMISMATCH,
|
2012-05-21 06:33:45 +00:00
|
|
|
MESHCMP_CDLAYERS_MISMATCH
|
2010-10-27 02:22:55 +00:00
|
|
|
};
|
|
|
|
|
2011-05-09 05:09:07 +00:00
|
|
|
static const char *cmpcode_to_str(int code)
|
2010-10-27 02:22:55 +00:00
|
|
|
{
|
|
|
|
switch (code) {
|
|
|
|
case MESHCMP_DVERT_WEIGHTMISMATCH:
|
|
|
|
return "Vertex Weight Mismatch";
|
|
|
|
case MESHCMP_DVERT_GROUPMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Vertex Group Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_DVERT_TOTGROUPMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Vertex Doesn't Belong To Same Number Of Groups";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_LOOPCOLMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Vertex Color Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_LOOPUVMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "UV Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_LOOPMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Loop Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_POLYVERTMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Loop Vert Mismatch In Poly Test";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_POLYMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Loop Vert Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_EDGEUNKNOWN:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Edge Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_VERTCOMISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Vertex Coordinate Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
case MESHCMP_CDLAYERS_MISMATCH:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "CustomData Layer Count Mismatch";
|
2010-10-27 02:22:55 +00:00
|
|
|
default:
|
2012-05-06 15:15:33 +00:00
|
|
|
return "Mesh Comparison Code Unknown";
|
|
|
|
}
|
2010-10-27 02:22:55 +00:00
|
|
|
}
|
|
|
|
|
2012-03-03 20:19:11 +00:00
|
|
|
/* thresh is threshold for comparing vertices, uvs, vertex colors,
|
|
|
|
* weights, etc.*/
|
2011-05-11 02:14:43 +00:00
|
|
|
static int customdata_compare(CustomData *c1, CustomData *c2, Mesh *m1, Mesh *m2, float thresh)
|
2010-10-27 02:22:55 +00:00
|
|
|
{
|
|
|
|
CustomDataLayer *l1, *l2;
|
2012-05-06 15:15:33 +00:00
|
|
|
int i, i1 = 0, i2 = 0, tot, j;
|
2010-10-27 02:22:55 +00:00
|
|
|
|
2012-04-28 06:31:57 +00:00
|
|
|
for (i = 0; i < c1->totlayer; i++) {
|
|
|
|
if (ELEM7(c1->layers[i].type, CD_MVERT, CD_MEDGE, CD_MPOLY,
|
|
|
|
CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT))
|
|
|
|
{
|
2010-10-27 02:22:55 +00:00
|
|
|
i1++;
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2010-10-27 02:22:55 +00:00
|
|
|
}
|
2012-04-28 06:31:57 +00:00
|
|
|
|
|
|
|
for (i = 0; i < c2->totlayer; i++) {
|
|
|
|
if (ELEM7(c2->layers[i].type, CD_MVERT, CD_MEDGE, CD_MPOLY,
|
|
|
|
CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT))
|
|
|
|
{
|
2010-10-27 02:22:55 +00:00
|
|
|
i2++;
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2010-10-27 02:22:55 +00:00
|
|
|
}
|
2012-04-28 06:31:57 +00:00
|
|
|
|
2010-10-27 02:22:55 +00:00
|
|
|
if (i1 != i2)
|
|
|
|
return MESHCMP_CDLAYERS_MISMATCH;
|
|
|
|
|
|
|
|
l1 = c1->layers; l2 = c2->layers;
|
|
|
|
tot = i1;
|
|
|
|
i1 = 0; i2 = 0;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < tot; i++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
while (i1 < c1->totlayer && !ELEM7(l1->type, CD_MVERT, CD_MEDGE, CD_MPOLY,
|
2012-04-28 06:31:57 +00:00
|
|
|
CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT))
|
|
|
|
{
|
2010-10-27 02:22:55 +00:00
|
|
|
i1++, l1++;
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2010-10-27 02:22:55 +00:00
|
|
|
|
2012-04-28 06:31:57 +00:00
|
|
|
while (i2 < c2->totlayer && !ELEM7(l2->type, CD_MVERT, CD_MEDGE, CD_MPOLY,
|
|
|
|
CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT))
|
|
|
|
{
|
2010-10-27 02:22:55 +00:00
|
|
|
i2++, l2++;
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2010-10-27 02:22:55 +00:00
|
|
|
|
|
|
|
if (l1->type == CD_MVERT) {
|
|
|
|
MVert *v1 = l1->data;
|
|
|
|
MVert *v2 = l2->data;
|
|
|
|
int vtot = m1->totvert;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < vtot; j++, v1++, v2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (len_v3v3(v1->co, v2->co) > thresh)
|
|
|
|
return MESHCMP_VERTCOMISMATCH;
|
2012-08-17 14:43:20 +00:00
|
|
|
/* I don't care about normals, let's just do coodinates */
|
2010-10-27 02:22:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*we're order-agnostic for edges here*/
|
|
|
|
if (l1->type == CD_MEDGE) {
|
|
|
|
MEdge *e1 = l1->data;
|
|
|
|
MEdge *e2 = l2->data;
|
|
|
|
EdgeHash *eh = BLI_edgehash_new();
|
|
|
|
int etot = m1->totedge;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < etot; j++, e1++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
BLI_edgehash_insert(eh, e1->v1, e1->v2, e1);
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < etot; j++, e2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (!BLI_edgehash_lookup(eh, e2->v1, e2->v2))
|
|
|
|
return MESHCMP_EDGEUNKNOWN;
|
|
|
|
}
|
|
|
|
BLI_edgehash_free(eh, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l1->type == CD_MPOLY) {
|
|
|
|
MPoly *p1 = l1->data;
|
|
|
|
MPoly *p2 = l2->data;
|
|
|
|
int ptot = m1->totpoly;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < ptot; j++, p1++, p2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
MLoop *lp1, *lp2;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
if (p1->totloop != p2->totloop)
|
|
|
|
return MESHCMP_POLYMISMATCH;
|
|
|
|
|
|
|
|
lp1 = m1->mloop + p1->loopstart;
|
|
|
|
lp2 = m2->mloop + p2->loopstart;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (k = 0; k < p1->totloop; k++, lp1++, lp2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (lp1->v != lp2->v)
|
|
|
|
return MESHCMP_POLYVERTMISMATCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (l1->type == CD_MLOOP) {
|
|
|
|
MLoop *lp1 = l1->data;
|
|
|
|
MLoop *lp2 = l2->data;
|
|
|
|
int ltot = m1->totloop;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < ltot; j++, lp1++, lp2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (lp1->v != lp2->v)
|
|
|
|
return MESHCMP_LOOPMISMATCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (l1->type == CD_MLOOPUV) {
|
|
|
|
MLoopUV *lp1 = l1->data;
|
|
|
|
MLoopUV *lp2 = l2->data;
|
|
|
|
int ltot = m1->totloop;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < ltot; j++, lp1++, lp2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (len_v2v2(lp1->uv, lp2->uv) > thresh)
|
|
|
|
return MESHCMP_LOOPUVMISMATCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l1->type == CD_MLOOPCOL) {
|
|
|
|
MLoopCol *lp1 = l1->data;
|
|
|
|
MLoopCol *lp2 = l2->data;
|
|
|
|
int ltot = m1->totloop;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < ltot; j++, lp1++, lp2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (ABS(lp1->r - lp2->r) > thresh ||
|
|
|
|
ABS(lp1->g - lp2->g) > thresh ||
|
|
|
|
ABS(lp1->b - lp2->b) > thresh ||
|
|
|
|
ABS(lp1->a - lp2->a) > thresh)
|
|
|
|
{
|
|
|
|
return MESHCMP_LOOPCOLMISMATCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l1->type == CD_MDEFORMVERT) {
|
|
|
|
MDeformVert *dv1 = l1->data;
|
|
|
|
MDeformVert *dv2 = l2->data;
|
|
|
|
int dvtot = m1->totvert;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < dvtot; j++, dv1++, dv2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
int k;
|
2012-05-06 15:15:33 +00:00
|
|
|
MDeformWeight *dw1 = dv1->dw, *dw2 = dv2->dw;
|
2010-10-27 02:22:55 +00:00
|
|
|
|
|
|
|
if (dv1->totweight != dv2->totweight)
|
|
|
|
return MESHCMP_DVERT_TOTGROUPMISMATCH;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (k = 0; k < dv1->totweight; k++, dw1++, dw2++) {
|
2010-10-27 02:22:55 +00:00
|
|
|
if (dw1->def_nr != dw2->def_nr)
|
|
|
|
return MESHCMP_DVERT_GROUPMISMATCH;
|
|
|
|
if (ABS(dw1->weight - dw2->weight) > thresh)
|
|
|
|
return MESHCMP_DVERT_WEIGHTMISMATCH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-11-11 23:36:56 +00:00
|
|
|
|
|
|
|
return 0;
|
2010-10-27 02:22:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*used for testing. returns an error string the two meshes don't match*/
|
2012-05-05 21:28:12 +00:00
|
|
|
const char *BKE_mesh_cmp(Mesh *me1, Mesh *me2, float thresh)
|
2010-10-27 02:22:55 +00:00
|
|
|
{
|
|
|
|
int c;
|
|
|
|
|
|
|
|
if (!me1 || !me2)
|
|
|
|
return "Requires two input meshes";
|
|
|
|
|
|
|
|
if (me1->totvert != me2->totvert)
|
|
|
|
return "Number of verts don't match";
|
|
|
|
|
|
|
|
if (me1->totedge != me2->totedge)
|
|
|
|
return "Number of edges don't match";
|
|
|
|
|
|
|
|
if (me1->totpoly != me2->totpoly)
|
|
|
|
return "Number of faces don't match";
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (me1->totloop != me2->totloop)
|
2010-10-27 02:22:55 +00:00
|
|
|
return "Number of loops don't match";
|
|
|
|
|
|
|
|
if ((c = customdata_compare(&me1->vdata, &me2->vdata, me1, me2, thresh)))
|
|
|
|
return cmpcode_to_str(c);
|
|
|
|
|
|
|
|
if ((c = customdata_compare(&me1->edata, &me2->edata, me1, me2, thresh)))
|
|
|
|
return cmpcode_to_str(c);
|
|
|
|
|
|
|
|
if ((c = customdata_compare(&me1->ldata, &me2->ldata, me1, me2, thresh)))
|
|
|
|
return cmpcode_to_str(c);
|
|
|
|
|
|
|
|
if ((c = customdata_compare(&me1->pdata, &me2->pdata, me1, me2, thresh)))
|
|
|
|
return cmpcode_to_str(c);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-03-02 16:05:54 +00:00
|
|
|
static void mesh_ensure_tessellation_customdata(Mesh *me)
|
2009-08-30 21:30:07 +00:00
|
|
|
{
|
2012-03-20 05:04:51 +00:00
|
|
|
if (UNLIKELY((me->totface != 0) && (me->totpoly == 0))) {
|
|
|
|
/* Pass, otherwise this function clears 'mface' before
|
|
|
|
* versioning 'mface -> mpoly' code kicks in [#30583]
|
|
|
|
*
|
|
|
|
* Callers could also check but safer to do here - campbell */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
const int tottex_original = CustomData_number_of_layers(&me->pdata, CD_MTEXPOLY);
|
|
|
|
const int totcol_original = CustomData_number_of_layers(&me->ldata, CD_MLOOPCOL);
|
2009-08-30 21:30:07 +00:00
|
|
|
|
2012-03-20 05:04:51 +00:00
|
|
|
const int tottex_tessface = CustomData_number_of_layers(&me->fdata, CD_MTFACE);
|
|
|
|
const int totcol_tessface = CustomData_number_of_layers(&me->fdata, CD_MCOL);
|
2011-12-06 09:28:25 +00:00
|
|
|
|
2012-03-20 05:04:51 +00:00
|
|
|
if (tottex_tessface != tottex_original ||
|
2012-05-06 15:15:33 +00:00
|
|
|
totcol_tessface != totcol_original)
|
2012-03-20 05:04:51 +00:00
|
|
|
{
|
|
|
|
BKE_mesh_tessface_clear(me);
|
2009-08-30 21:30:07 +00:00
|
|
|
|
2012-03-20 05:04:51 +00:00
|
|
|
CustomData_from_bmeshpoly(&me->fdata, &me->pdata, &me->ldata, me->totface);
|
2011-12-06 09:28:25 +00:00
|
|
|
|
2012-04-12 14:36:57 +00:00
|
|
|
/* TODO - add some --debug-mesh option */
|
|
|
|
if (G.debug & G_DEBUG) {
|
2012-04-25 06:06:40 +00:00
|
|
|
/* note: this warning may be un-called for if we are initializing the mesh for the
|
2012-04-12 14:36:57 +00:00
|
|
|
* first time from bmesh, rather then giving a warning about this we could be smarter
|
|
|
|
* and check if there was any data to begin with, for now just print the warning with
|
|
|
|
* some info to help troubleshoot whats going on - campbell */
|
|
|
|
printf("%s: warning! Tessellation uvs or vcol data got out of sync, "
|
2012-05-06 15:15:33 +00:00
|
|
|
"had to reset!\n CD_MTFACE: %d != CD_MTEXPOLY: %d || CD_MCOL: %d != CD_MLOOPCOL: %d\n",
|
|
|
|
__func__, tottex_tessface, tottex_original, totcol_tessface, totcol_original);
|
2012-04-12 14:36:57 +00:00
|
|
|
}
|
2012-03-20 05:04:51 +00:00
|
|
|
}
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-06 09:28:25 +00:00
|
|
|
/* this ensures grouped customdata (e.g. mtexpoly and mloopuv and mtface, or
|
|
|
|
* mloopcol and mcol) have the same relative active/render/clone/mask indices.
|
|
|
|
*
|
|
|
|
* note that for undo mesh data we want to skip 'ensure_tess_cd' call since
|
2012-03-18 07:38:51 +00:00
|
|
|
* we don't want to store memory for tessface when its only used for older
|
2011-12-06 09:28:25 +00:00
|
|
|
* versions of the mesh. - campbell*/
|
2013-03-17 19:55:10 +00:00
|
|
|
static void mesh_update_linked_customdata(Mesh *me, const bool do_ensure_tess_cd)
|
2009-08-30 21:30:07 +00:00
|
|
|
{
|
|
|
|
if (me->edit_btmesh)
|
|
|
|
BMEdit_UpdateLinkedCustomData(me->edit_btmesh);
|
|
|
|
|
2011-12-06 09:28:25 +00:00
|
|
|
if (do_ensure_tess_cd) {
|
2012-03-02 16:05:54 +00:00
|
|
|
mesh_ensure_tessellation_customdata(me);
|
2011-12-06 09:28:25 +00:00
|
|
|
}
|
2009-08-30 21:30:07 +00:00
|
|
|
|
2011-12-01 09:49:27 +00:00
|
|
|
CustomData_bmesh_update_active_layers(&me->fdata, &me->pdata, &me->ldata);
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
2009-05-23 03:24:15 +00:00
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
void BKE_mesh_update_customdata_pointers(Mesh *me, const bool do_ensure_tess_cd)
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
{
|
2011-12-06 09:28:25 +00:00
|
|
|
mesh_update_linked_customdata(me, do_ensure_tess_cd);
|
2009-08-30 21:30:07 +00:00
|
|
|
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
me->mvert = CustomData_get_layer(&me->vdata, CD_MVERT);
|
|
|
|
me->dvert = CustomData_get_layer(&me->vdata, CD_MDEFORMVERT);
|
|
|
|
|
|
|
|
me->medge = CustomData_get_layer(&me->edata, CD_MEDGE);
|
|
|
|
|
|
|
|
me->mface = CustomData_get_layer(&me->fdata, CD_MFACE);
|
|
|
|
me->mcol = CustomData_get_layer(&me->fdata, CD_MCOL);
|
|
|
|
me->mtface = CustomData_get_layer(&me->fdata, CD_MTFACE);
|
2009-07-16 06:27:37 +00:00
|
|
|
|
2009-05-26 04:17:47 +00:00
|
|
|
me->mpoly = CustomData_get_layer(&me->pdata, CD_MPOLY);
|
|
|
|
me->mloop = CustomData_get_layer(&me->ldata, CD_MLOOP);
|
|
|
|
|
|
|
|
me->mtpoly = CustomData_get_layer(&me->pdata, CD_MTEXPOLY);
|
|
|
|
me->mloopcol = CustomData_get_layer(&me->ldata, CD_MLOOPCOL);
|
|
|
|
me->mloopuv = CustomData_get_layer(&me->ldata, CD_MLOOPUV);
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
}
|
|
|
|
|
2006-09-12 14:05:26 +00:00
|
|
|
/* Note: unlinking is called when me->id.us is 0, question remains how
|
|
|
|
* much unlinking of Library data in Mesh should be done... probably
|
|
|
|
* we need a more generic method, like the expand() functions in
|
|
|
|
* readfile.c */
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_unlink(Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
int a;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (me == NULL) return;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-07-31 13:43:26 +00:00
|
|
|
if (me->mat)
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < me->totcol; a++) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->mat[a]) me->mat[a]->id.us--;
|
2012-05-06 15:15:33 +00:00
|
|
|
me->mat[a] = NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2006-09-12 14:05:26 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->key) {
|
2011-12-04 06:39:35 +00:00
|
|
|
me->key->id.us--;
|
2006-09-12 14:05:26 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
me->key = NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (me->texcomesh) me->texcomesh = NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2003-04-26 13:07:59 +00:00
|
|
|
/* do not free mesh itself */
|
2012-05-05 14:03:12 +00:00
|
|
|
void BKE_mesh_free(Mesh *me, int unlink)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2009-09-10 06:08:52 +00:00
|
|
|
if (unlink)
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_unlink(me);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
CustomData_free(&me->vdata, me->totvert);
|
|
|
|
CustomData_free(&me->edata, me->totedge);
|
|
|
|
CustomData_free(&me->fdata, me->totface);
|
2009-05-26 04:17:47 +00:00
|
|
|
CustomData_free(&me->ldata, me->totloop);
|
|
|
|
CustomData_free(&me->pdata, me->totpoly);
|
2004-07-08 20:38:27 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->adt) {
|
2009-12-28 00:52:31 +00:00
|
|
|
BKE_free_animdata(&me->id);
|
2012-05-06 15:15:33 +00:00
|
|
|
me->adt = NULL;
|
2009-12-28 00:52:31 +00:00
|
|
|
}
|
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->mat) MEM_freeN(me->mat);
|
2005-07-18 18:28:16 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->bb) MEM_freeN(me->bb);
|
|
|
|
if (me->mselect) MEM_freeN(me->mselect);
|
|
|
|
if (me->edit_btmesh) MEM_freeN(me->edit_btmesh);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-03-06 16:22:41 +00:00
|
|
|
static void mesh_tessface_clear_intern(Mesh *mesh, int free_customdata)
|
|
|
|
{
|
2012-10-31 09:50:24 +00:00
|
|
|
if (free_customdata) {
|
2012-03-06 16:22:41 +00:00
|
|
|
CustomData_free(&mesh->fdata, mesh->totface);
|
2012-10-31 09:50:24 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
CustomData_reset(&mesh->fdata);
|
|
|
|
}
|
2012-03-06 16:22:41 +00:00
|
|
|
|
|
|
|
mesh->mface = NULL;
|
|
|
|
mesh->mtface = NULL;
|
|
|
|
mesh->mcol = NULL;
|
|
|
|
mesh->totface = 0;
|
|
|
|
}
|
|
|
|
|
2013-02-05 12:46:15 +00:00
|
|
|
Mesh *BKE_mesh_add(Main *bmain, const char *name)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
Mesh *me;
|
|
|
|
|
2013-02-05 12:46:15 +00:00
|
|
|
me = BKE_libblock_alloc(&bmain->mesh, ID_ME, name);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
me->size[0] = me->size[1] = me->size[2] = 1.0;
|
|
|
|
me->smoothresh = 30;
|
|
|
|
me->texflag = ME_AUTOSPACE;
|
|
|
|
me->flag = ME_TWOSIDED;
|
|
|
|
me->drawflag = ME_DRAWEDGES | ME_DRAWFACES | ME_DRAWCREASES;
|
2012-10-31 15:14:14 +00:00
|
|
|
|
|
|
|
CustomData_reset(&me->vdata);
|
|
|
|
CustomData_reset(&me->edata);
|
|
|
|
CustomData_reset(&me->fdata);
|
|
|
|
CustomData_reset(&me->pdata);
|
|
|
|
CustomData_reset(&me->ldata);
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
return me;
|
|
|
|
}
|
|
|
|
|
2013-02-05 12:46:15 +00:00
|
|
|
Mesh *BKE_mesh_copy_ex(Main *bmain, Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
Mesh *men;
|
2006-12-12 21:29:09 +00:00
|
|
|
MTFace *tface;
|
2009-05-26 04:17:47 +00:00
|
|
|
MTexPoly *txface;
|
2006-12-12 21:29:09 +00:00
|
|
|
int a, i;
|
2012-03-05 18:05:06 +00:00
|
|
|
const int do_tessface = ((me->totface != 0) && (me->totpoly == 0)); /* only do tessface if we have no polys */
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2013-02-05 12:46:15 +00:00
|
|
|
men = BKE_libblock_copy_ex(bmain, &me->id);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
men->mat = MEM_dupallocN(me->mat);
|
|
|
|
for (a = 0; a < men->totcol; a++) {
|
2002-10-12 11:37:38 +00:00
|
|
|
id_us_plus((ID *)men->mat[a]);
|
|
|
|
}
|
|
|
|
id_us_plus((ID *)men->texcomesh);
|
|
|
|
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
CustomData_copy(&me->vdata, &men->vdata, CD_MASK_MESH, CD_DUPLICATE, men->totvert);
|
|
|
|
CustomData_copy(&me->edata, &men->edata, CD_MASK_MESH, CD_DUPLICATE, men->totedge);
|
2009-05-26 04:17:47 +00:00
|
|
|
CustomData_copy(&me->ldata, &men->ldata, CD_MASK_MESH, CD_DUPLICATE, men->totloop);
|
|
|
|
CustomData_copy(&me->pdata, &men->pdata, CD_MASK_MESH, CD_DUPLICATE, men->totpoly);
|
2012-03-05 18:05:06 +00:00
|
|
|
if (do_tessface) {
|
|
|
|
CustomData_copy(&me->fdata, &men->fdata, CD_MASK_MESH, CD_DUPLICATE, men->totface);
|
|
|
|
}
|
|
|
|
else {
|
2012-03-06 16:22:41 +00:00
|
|
|
mesh_tessface_clear_intern(men, FALSE);
|
2012-03-05 18:05:06 +00:00
|
|
|
}
|
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
BKE_mesh_update_customdata_pointers(men, do_tessface);
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
/* ensure indirect linked data becomes lib-extern */
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->fdata.totlayer; i++) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->fdata.layers[i].type == CD_MTFACE) {
|
2012-05-06 15:15:33 +00:00
|
|
|
tface = (MTFace *)me->fdata.layers[i].data;
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < me->totface; a++, tface++)
|
2012-02-23 02:17:50 +00:00
|
|
|
if (tface->tpage)
|
2012-05-06 15:15:33 +00:00
|
|
|
id_lib_extern((ID *)tface->tpage);
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
2006-10-16 11:31:09 +00:00
|
|
|
}
|
2006-12-01 18:41:25 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->pdata.totlayer; i++) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->pdata.layers[i].type == CD_MTEXPOLY) {
|
2012-05-06 15:15:33 +00:00
|
|
|
txface = (MTexPoly *)me->pdata.layers[i].data;
|
2009-05-26 04:17:47 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < me->totpoly; a++, txface++)
|
2012-02-23 02:17:50 +00:00
|
|
|
if (txface->tpage)
|
2012-05-06 15:15:33 +00:00
|
|
|
id_lib_extern((ID *)txface->tpage);
|
2009-05-26 04:17:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
men->mselect = NULL;
|
|
|
|
men->edit_btmesh = NULL;
|
2006-05-10 17:22:49 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
men->bb = MEM_dupallocN(men->bb);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
men->key = BKE_key_copy(me->key);
|
|
|
|
if (men->key) men->key->from = (ID *)men;
|
2006-08-20 15:22:56 +00:00
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
return men;
|
|
|
|
}
|
|
|
|
|
2013-02-05 12:46:15 +00:00
|
|
|
Mesh *BKE_mesh_copy(Mesh *me)
|
|
|
|
{
|
|
|
|
return BKE_mesh_copy_ex(G.main, me);
|
|
|
|
}
|
|
|
|
|
2011-09-12 02:23:30 +00:00
|
|
|
BMesh *BKE_mesh_to_bmesh(Mesh *me, Object *ob)
|
2009-05-26 04:17:47 +00:00
|
|
|
{
|
|
|
|
BMesh *bm;
|
|
|
|
|
2012-03-11 19:58:56 +00:00
|
|
|
bm = BM_mesh_create(&bm_mesh_allocsize_default);
|
2009-05-26 04:17:47 +00:00
|
|
|
|
2013-01-14 16:42:43 +00:00
|
|
|
BM_mesh_bm_from_me(bm, me, true, ob->shapenr);
|
2009-05-26 04:17:47 +00:00
|
|
|
|
|
|
|
return bm;
|
|
|
|
}
|
|
|
|
|
2011-10-23 17:52:20 +00:00
|
|
|
static void expand_local_mesh(Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2011-10-23 17:52:20 +00:00
|
|
|
id_lib_extern((ID *)me->texcomesh);
|
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->mtface || me->mtpoly) {
|
2011-10-23 17:52:20 +00:00
|
|
|
int a, i;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->pdata.totlayer; i++) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->pdata.layers[i].type == CD_MTEXPOLY) {
|
2012-10-11 00:56:34 +00:00
|
|
|
MTexPoly *txface = (MTexPoly *)me->pdata.layers[i].data;
|
2011-10-27 09:42:03 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < me->totpoly; a++, txface++) {
|
2011-10-27 09:42:03 +00:00
|
|
|
/* special case: ima always local immediately */
|
2012-02-23 02:17:50 +00:00
|
|
|
if (txface->tpage) {
|
2012-03-08 04:38:35 +00:00
|
|
|
id_lib_extern((ID *)txface->tpage);
|
2009-05-26 04:17:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->fdata.totlayer; i++) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->fdata.layers[i].type == CD_MTFACE) {
|
2012-05-06 15:15:33 +00:00
|
|
|
MTFace *tface = (MTFace *)me->fdata.layers[i].data;
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < me->totface; a++, tface++) {
|
2011-10-27 09:42:03 +00:00
|
|
|
/* special case: ima always local immediately */
|
2012-02-23 02:17:50 +00:00
|
|
|
if (tface->tpage) {
|
2012-03-08 04:38:35 +00:00
|
|
|
id_lib_extern((ID *)tface->tpage);
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-26 07:17:21 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->mat) {
|
2011-04-26 07:17:21 +00:00
|
|
|
extern_local_matarar(me->mat, me->totcol);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-05 14:03:12 +00:00
|
|
|
void BKE_mesh_make_local(Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
Main *bmain = G.main;
|
2002-10-12 11:37:38 +00:00
|
|
|
Object *ob;
|
2012-05-06 15:15:33 +00:00
|
|
|
int is_local = FALSE, is_lib = FALSE;
|
2003-04-26 13:07:59 +00:00
|
|
|
|
|
|
|
/* - only lib users: do nothing
|
2011-04-26 07:17:21 +00:00
|
|
|
* - only local users: set flag
|
|
|
|
* - mixed: make copy
|
|
|
|
*/
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (me->id.lib == NULL) return;
|
|
|
|
if (me->id.us == 1) {
|
2011-12-28 22:47:55 +00:00
|
|
|
id_clear_lib_data(bmain, &me->id);
|
2011-10-23 17:52:20 +00:00
|
|
|
expand_local_mesh(me);
|
2002-10-12 11:37:38 +00:00
|
|
|
return;
|
|
|
|
}
|
2011-04-26 07:17:21 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (ob = bmain->object.first; ob && ELEM(0, is_lib, is_local); ob = ob->id.next) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me == ob->data) {
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ob->id.lib) is_lib = TRUE;
|
|
|
|
else is_local = TRUE;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-04-26 07:17:21 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (is_local && is_lib == FALSE) {
|
2011-12-28 22:47:55 +00:00
|
|
|
id_clear_lib_data(bmain, &me->id);
|
2011-10-23 17:52:20 +00:00
|
|
|
expand_local_mesh(me);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-02-23 02:17:50 +00:00
|
|
|
else if (is_local && is_lib) {
|
2012-05-06 15:15:33 +00:00
|
|
|
Mesh *me_new = BKE_mesh_copy(me);
|
|
|
|
me_new->id.us = 0;
|
2011-04-26 07:17:21 +00:00
|
|
|
|
2011-12-28 22:47:55 +00:00
|
|
|
|
|
|
|
/* Remap paths of new ID using old library as base. */
|
2011-11-30 00:32:13 +00:00
|
|
|
BKE_id_lib_local_paths(bmain, me->id.lib, &me_new->id);
|
2011-10-27 05:34:39 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (ob = bmain->object.first; ob; ob = ob->id.next) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me == ob->data) {
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ob->id.lib == NULL) {
|
2013-03-17 19:55:10 +00:00
|
|
|
BKE_mesh_assign_object(ob, me_new);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_boundbox_calc(Mesh *me, float r_loc[3], float r_size[3])
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
BoundBox *bb;
|
|
|
|
float min[3], max[3];
|
|
|
|
float mloc[3], msize[3];
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (me->bb == NULL) me->bb = MEM_callocN(sizeof(BoundBox), "boundbox");
|
|
|
|
bb = me->bb;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (!r_loc) r_loc = mloc;
|
|
|
|
if (!r_size) r_size = msize;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2010-08-03 00:56:43 +00:00
|
|
|
INIT_MINMAX(min, max);
|
2012-05-05 21:28:12 +00:00
|
|
|
if (!BKE_mesh_minmax(me, min, max)) {
|
2005-07-18 17:33:51 +00:00
|
|
|
min[0] = min[1] = min[2] = -1.0f;
|
|
|
|
max[0] = max[1] = max[2] = 1.0f;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
mid_v3_v3v3(r_loc, min, max);
|
2005-07-18 17:33:51 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
r_size[0] = (max[0] - min[0]) / 2.0f;
|
|
|
|
r_size[1] = (max[1] - min[1]) / 2.0f;
|
|
|
|
r_size[2] = (max[2] - min[2]) / 2.0f;
|
2005-07-18 17:33:51 +00:00
|
|
|
|
2012-05-05 14:03:12 +00:00
|
|
|
BKE_boundbox_init_from_minmax(bb, min, max);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_texspace_calc(Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2009-11-01 00:06:53 +00:00
|
|
|
float loc[3], size[3];
|
2002-10-12 11:37:38 +00:00
|
|
|
int a;
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_boundbox_calc(me, loc, size);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-02-19 22:17:30 +00:00
|
|
|
if (me->texflag & ME_AUTOSPACE) {
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < 3; a++) {
|
|
|
|
if (size[a] == 0.0f) size[a] = 1.0f;
|
|
|
|
else if (size[a] > 0.0f && size[a] < 0.00001f) size[a] = 0.00001f;
|
|
|
|
else if (size[a] < 0.0f && size[a] > -0.00001f) size[a] = -0.00001f;
|
2005-07-14 21:57:18 +00:00
|
|
|
}
|
|
|
|
|
2010-08-01 11:00:36 +00:00
|
|
|
copy_v3_v3(me->loc, loc);
|
|
|
|
copy_v3_v3(me->size, size);
|
|
|
|
zero_v3(me->rot);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BoundBox *BKE_mesh_boundbox_get(Object *ob)
|
2005-07-14 18:04:27 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
Mesh *me = ob->data;
|
2007-12-05 21:50:23 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (ob->bb)
|
2007-12-05 21:50:23 +00:00
|
|
|
return ob->bb;
|
|
|
|
|
|
|
|
if (!me->bb)
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_texspace_calc(me);
|
2005-07-14 18:04:27 +00:00
|
|
|
|
|
|
|
return me->bb;
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_texspace_get(Mesh *me, float r_loc[3], float r_rot[3], float r_size[3])
|
2005-07-14 21:57:18 +00:00
|
|
|
{
|
|
|
|
if (!me->bb) {
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_texspace_calc(me);
|
2005-07-14 21:57:18 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (r_loc) copy_v3_v3(r_loc, me->loc);
|
|
|
|
if (r_rot) copy_v3_v3(r_rot, me->rot);
|
2012-02-28 14:05:00 +00:00
|
|
|
if (r_size) copy_v3_v3(r_size, me->size);
|
2005-07-14 21:57:18 +00:00
|
|
|
}
|
|
|
|
|
2012-12-21 07:28:14 +00:00
|
|
|
float (*BKE_mesh_orco_verts_get(Object *ob))[3]
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2005-07-20 04:44:02 +00:00
|
|
|
Mesh *me = ob->data;
|
2009-11-01 00:06:53 +00:00
|
|
|
MVert *mvert = NULL;
|
2012-05-06 15:15:33 +00:00
|
|
|
Mesh *tme = me->texcomesh ? me->texcomesh : me;
|
2005-07-18 19:58:23 +00:00
|
|
|
int a, totvert;
|
2007-01-21 23:46:00 +00:00
|
|
|
float (*vcos)[3] = NULL;
|
2005-07-18 18:28:16 +00:00
|
|
|
|
2007-12-05 12:40:54 +00:00
|
|
|
/* Get appropriate vertex coordinates */
|
2012-05-06 15:15:33 +00:00
|
|
|
vcos = MEM_callocN(sizeof(*vcos) * me->totvert, "orco mesh");
|
2009-11-01 00:06:53 +00:00
|
|
|
mvert = tme->mvert;
|
2012-10-23 13:28:22 +00:00
|
|
|
totvert = min_ii(tme->totvert, me->totvert);
|
2009-11-01 00:06:53 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < totvert; a++, mvert++) {
|
2010-08-01 11:00:36 +00:00
|
|
|
copy_v3_v3(vcos[a], mvert->co);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2005-07-18 18:28:16 +00:00
|
|
|
|
2012-12-21 07:28:14 +00:00
|
|
|
return vcos;
|
2007-12-05 12:40:54 +00:00
|
|
|
}
|
2005-07-18 19:58:23 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_orco_verts_transform(Mesh *me, float (*orco)[3], int totvert, int invert)
|
2007-12-05 12:40:54 +00:00
|
|
|
{
|
|
|
|
float loc[3], size[3];
|
|
|
|
int a;
|
2005-07-18 19:58:23 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_texspace_get(me->texcomesh ? me->texcomesh : me, loc, NULL, size);
|
2005-07-18 19:58:23 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (invert) {
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < totvert; a++) {
|
2007-12-11 20:02:21 +00:00
|
|
|
float *co = orco[a];
|
2010-08-01 11:00:36 +00:00
|
|
|
madd_v3_v3v3v3(co, loc, co, size);
|
2007-12-11 20:02:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < totvert; a++) {
|
2007-12-11 20:02:21 +00:00
|
|
|
float *co = orco[a];
|
2012-05-06 15:15:33 +00:00
|
|
|
co[0] = (co[0] - loc[0]) / size[0];
|
|
|
|
co[1] = (co[1] - loc[1]) / size[1];
|
|
|
|
co[2] = (co[2] - loc[2]) / size[2];
|
2007-12-11 20:02:21 +00:00
|
|
|
}
|
2005-07-18 19:58:23 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2006-11-08 20:14:04 +00:00
|
|
|
/* rotates the vertices of a face in case v[2] or v[3] (vertex index) is = 0.
|
2012-03-03 20:19:11 +00:00
|
|
|
* this is necessary to make the if (mface->v4) check for quads work */
|
2008-01-29 19:49:03 +00:00
|
|
|
int test_index_face(MFace *mface, CustomData *fdata, int mfindex, int nr)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
/* first test if the face is legal */
|
2012-05-06 15:15:33 +00:00
|
|
|
if ((mface->v3 || nr == 4) && mface->v3 == mface->v4) {
|
|
|
|
mface->v4 = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
nr--;
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
if ((mface->v2 || mface->v4) && mface->v2 == mface->v3) {
|
|
|
|
mface->v3 = mface->v4;
|
|
|
|
mface->v4 = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
nr--;
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
if (mface->v1 == mface->v2) {
|
|
|
|
mface->v2 = mface->v3;
|
|
|
|
mface->v3 = mface->v4;
|
|
|
|
mface->v4 = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
nr--;
|
|
|
|
}
|
|
|
|
|
2012-09-26 20:05:38 +00:00
|
|
|
/* check corrupt cases, bow-tie geometry, cant handle these because edge data wont exist so just return 0 */
|
2012-05-06 15:15:33 +00:00
|
|
|
if (nr == 3) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (
|
2012-05-06 15:15:33 +00:00
|
|
|
/* real edges */
|
|
|
|
mface->v1 == mface->v2 ||
|
|
|
|
mface->v2 == mface->v3 ||
|
|
|
|
mface->v3 == mface->v1)
|
|
|
|
{
|
2011-02-09 04:45:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (nr == 4) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (
|
2012-05-06 15:15:33 +00:00
|
|
|
/* real edges */
|
|
|
|
mface->v1 == mface->v2 ||
|
|
|
|
mface->v2 == mface->v3 ||
|
|
|
|
mface->v3 == mface->v4 ||
|
|
|
|
mface->v4 == mface->v1 ||
|
|
|
|
/* across the face */
|
|
|
|
mface->v1 == mface->v3 ||
|
2012-05-20 19:49:27 +00:00
|
|
|
mface->v2 == mface->v4)
|
|
|
|
{
|
2011-02-09 04:45:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-04-26 13:07:59 +00:00
|
|
|
/* prevent a zero at wrong index location */
|
2012-05-06 15:15:33 +00:00
|
|
|
if (nr == 3) {
|
|
|
|
if (mface->v3 == 0) {
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
static int corner_indices[4] = {1, 2, 0, 3};
|
|
|
|
|
2011-12-27 03:54:23 +00:00
|
|
|
SWAP(unsigned int, mface->v1, mface->v2);
|
|
|
|
SWAP(unsigned int, mface->v2, mface->v3);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (fdata)
|
2006-11-20 19:03:37 +00:00
|
|
|
CustomData_swap(fdata, mfindex, corner_indices);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (nr == 4) {
|
|
|
|
if (mface->v3 == 0 || mface->v4 == 0) {
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
static int corner_indices[4] = {2, 3, 0, 1};
|
|
|
|
|
2011-12-27 03:54:23 +00:00
|
|
|
SWAP(unsigned int, mface->v1, mface->v3);
|
|
|
|
SWAP(unsigned int, mface->v2, mface->v4);
|
- convert all DerivedMesh map functions to use index based
mapping (instead of Edit{Vert,Edge,Face} pointers)
- dropped convertToDispListMeshMapped (whew, glad of it too)
- added DerivedMesh drawMappedFaces function
- dropped EM suffix for DerivedMesh functions, it was neither
particularly correct nor descriptive
- converted test_index_mface to test_index_face that also corrects
MCol and TFace. Good thing we had three versions of this routine,
you never know when one might burn down.
- removed flipnorm_mesh, not used anymore (and was incorrect to
boot)
- Getting face select to work with modifiers turned out to be much
more complicated than expected. Reworked mapping architecture for
modifiers - basically elements in a DispListMesh are now required
to be stored in an order that corresponds exactly to original
ordering. MVert/MEdge/MFace all have a new flag ME_XXX_STEPINDEX
that is set on each element that is set on the first derived element
of each original element. I can't say the code to follow these
requirements for subsurf is particularly transparent, but on the
upside it is a reasonably consistent and simple system that is memory
efficient and allows keeping the DispListMesh structure.
- rewrote mirror modifier to be simpler/conform to new requirements
for mapped DispListMesh structure. This also means that mirror interacts
much better with incremental subsurf calculation (it used to recalc
one entire side on any topology change, now it generally avoids that).
- added EM_{init,free}_index_arrays and EM_get_{vert,edge,face}_for_index
functions to handle mapping indices back into appropriate EditMesh
structures.
- bug fix, make edges didn't recalc object data
- bug fix, initial image assignment to TFace's didn't recalc object data
- new feature, added circle select support for FACESELECT
- bug fix, creating new faces in editmode duplicated the TFACE active
flag - but there should only be one active tface
- bug fix, possible crash when deleting all faces in faceselect mode
on mesh with tfaces...
Still todo: TFace edge drawing is still not always correct in face
mode, in particular with a mirror modifier when mesh has edges (and
no preceeding subsurf). Have not yet decided how to deal with this.
Best solution is probably to do switch to meshes all having MEdge's,
in which case I can get rid of TFace edge flags (and need to recalc
modifiers on tface selection change).
2005-08-20 03:08:23 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (fdata)
|
2006-11-20 19:03:37 +00:00
|
|
|
CustomData_swap(fdata, mfindex, corner_indices);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
2008-01-29 19:49:03 +00:00
|
|
|
|
|
|
|
return nr;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 16:03:57 +00:00
|
|
|
Mesh *BKE_mesh_from_object(Object *ob)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ob == NULL) return NULL;
|
|
|
|
if (ob->type == OB_MESH) return ob->data;
|
2011-02-13 10:52:18 +00:00
|
|
|
else return NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
void BKE_mesh_assign_object(Object *ob, Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
Mesh *old = NULL;
|
2010-09-09 00:14:51 +00:00
|
|
|
|
|
|
|
multires_force_update(ob);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ob == NULL) return;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ob->type == OB_MESH) {
|
|
|
|
old = ob->data;
|
2009-10-22 23:22:05 +00:00
|
|
|
if (old)
|
|
|
|
old->id.us--;
|
2012-05-06 15:15:33 +00:00
|
|
|
ob->data = me;
|
2002-10-12 11:37:38 +00:00
|
|
|
id_us_plus((ID *)me);
|
|
|
|
}
|
|
|
|
|
|
|
|
test_object_materials((ID *)me);
|
2010-09-09 00:14:51 +00:00
|
|
|
|
|
|
|
test_object_modifiers(ob);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2004-07-08 20:38:27 +00:00
|
|
|
/* ************** make edges in a Mesh, for outside of editmode */
|
|
|
|
|
|
|
|
struct edgesort {
|
2011-12-27 03:54:23 +00:00
|
|
|
unsigned int v1, v2;
|
2005-09-22 17:00:58 +00:00
|
|
|
short is_loose, is_draw;
|
2004-07-08 20:38:27 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* edges have to be added with lowest index first for sorting */
|
2011-12-27 03:54:23 +00:00
|
|
|
static void to_edgesort(struct edgesort *ed,
|
|
|
|
unsigned int v1, unsigned int v2,
|
|
|
|
short is_loose, short is_draw)
|
2004-07-08 20:38:27 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
if (v1 < v2) {
|
|
|
|
ed->v1 = v1; ed->v2 = v2;
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-05-06 15:15:33 +00:00
|
|
|
ed->v1 = v2; ed->v2 = v1;
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
ed->is_loose = is_loose;
|
|
|
|
ed->is_draw = is_draw;
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vergedgesort(const void *v1, const void *v2)
|
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
const struct edgesort *x1 = v1, *x2 = v2;
|
2004-07-08 20:38:27 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (x1->v1 > x2->v1) return 1;
|
|
|
|
else if (x1->v1 < x2->v1) return -1;
|
|
|
|
else if (x1->v2 > x2->v2) return 1;
|
|
|
|
else if (x1->v2 < x2->v2) return -1;
|
2004-07-08 20:38:27 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-02-03 04:58:55 +00:00
|
|
|
|
2010-03-05 16:47:52 +00:00
|
|
|
/* Create edges based on known verts and faces */
|
2011-04-15 01:19:13 +00:00
|
|
|
static void make_edges_mdata(MVert *UNUSED(allvert), MFace *allface, MLoop *allloop,
|
2012-05-06 15:15:33 +00:00
|
|
|
MPoly *allpoly, int UNUSED(totvert), int totface, int UNUSED(totloop), int totpoly,
|
|
|
|
int old, MEdge **alledge, int *_totedge)
|
2004-07-08 20:38:27 +00:00
|
|
|
{
|
2011-04-15 01:19:13 +00:00
|
|
|
MPoly *mpoly;
|
2004-07-08 20:38:27 +00:00
|
|
|
MFace *mface;
|
|
|
|
MEdge *medge;
|
2011-04-15 01:19:13 +00:00
|
|
|
EdgeHash *hash = BLI_edgehash_new();
|
2004-07-08 20:38:27 +00:00
|
|
|
struct edgesort *edsort, *ed;
|
2013-02-28 04:18:01 +00:00
|
|
|
int a, totedge = 0, final = 0;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2004-07-08 20:38:27 +00:00
|
|
|
/* we put all edges in array, sort them, and detect doubles that way */
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = totface, mface = allface; a > 0; a--, mface++) {
|
|
|
|
if (mface->v4) totedge += 4;
|
|
|
|
else if (mface->v3) totedge += 3;
|
|
|
|
else totedge += 1;
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (totedge == 0) {
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
/* flag that mesh has edges */
|
2012-05-06 15:15:33 +00:00
|
|
|
(*alledge) = MEM_callocN(0, "make mesh edges");
|
2010-03-05 16:47:52 +00:00
|
|
|
(*_totedge) = 0;
|
- added data arguments to deformer modifiers, in case someone wants
to write one that is based on geometry (and not just vertex position)
- added editmode versions of modifier deform/apply calls and flag
to tag modifiers that support editmode
- added isFinalCalc param to applyModifier, basically a switch to let
subsurf know if it is calc'ng orco or not (so it can deal with cache
appropriately). This is kinda hacky and perhaps I can come up with
a better solution (its also a waste to do a complete subdivide just
to get vertex locations).
- changed ccgsubsurf to not preallocate hash's to be approximately correct
size... this was probably not a big performance savings but means that
the order of faces returned by the iterator can vary after the first
call, this messes up orco calculation so dropped for time being.
- minor bug fix, meshes with only key didn't get vertex normals correctly
calc'd
- updated editmesh derivedmesh to support auxiliary locations
- changed mesh_calc_modifiers to alloc deformVerts on demand
- added editmesh_calc_modifiers for calculating editmesh cage and final
derivedmesh's
- bug fix, update shadedisplist to always calc colors (even if totvert==0)
- changed load_editMesh and make_edge to build me->medge even if totedge==0
(incremental subsurf checks this)
todo: add drawFacesTex for ccgderivedmesh
So, modifiers in editmode are back (which means auto-mirror
in edit mode works now) although still not finished. Currently
no cage is computed, the cage is always the base mesh (in
other words, Optimal edge style editing is off), and the final
mesh currently includes all modifiers that work in edit mode
(including lattice and curve). At some point there will be toggles
for which modifiers affect the final/cage editmode derivedmesh's.
Also, very nice new feature is that incremental subsurf in object
mode returns a ccgderivedmesh object instead of copying to a new
displistmesh. This can make a *huge* speed difference, and is very
nice for working with deformed armatures (esp. with only small
per frame changes).
2005-07-22 07:37:15 +00:00
|
|
|
return;
|
|
|
|
}
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
ed = edsort = MEM_mallocN(totedge * sizeof(struct edgesort), "edgesort");
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = totface, mface = allface; a > 0; a--, mface++) {
|
2005-09-22 17:00:58 +00:00
|
|
|
to_edgesort(ed++, mface->v1, mface->v2, !mface->v3, mface->edcode & ME_V1V2);
|
2012-02-23 02:17:50 +00:00
|
|
|
if (mface->v4) {
|
2005-09-22 17:00:58 +00:00
|
|
|
to_edgesort(ed++, mface->v2, mface->v3, 0, mface->edcode & ME_V2V3);
|
|
|
|
to_edgesort(ed++, mface->v3, mface->v4, 0, mface->edcode & ME_V3V4);
|
|
|
|
to_edgesort(ed++, mface->v4, mface->v1, 0, mface->edcode & ME_V4V1);
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
2012-02-23 02:17:50 +00:00
|
|
|
else if (mface->v3) {
|
2005-09-22 17:00:58 +00:00
|
|
|
to_edgesort(ed++, mface->v2, mface->v3, 0, mface->edcode & ME_V2V3);
|
|
|
|
to_edgesort(ed++, mface->v3, mface->v1, 0, mface->edcode & ME_V3V1);
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
|
|
|
}
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2004-07-08 20:38:27 +00:00
|
|
|
qsort(edsort, totedge, sizeof(struct edgesort), vergedgesort);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2004-07-08 20:38:27 +00:00
|
|
|
/* count final amount */
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = totedge, ed = edsort; a > 1; a--, ed++) {
|
2004-07-08 20:38:27 +00:00
|
|
|
/* edge is unique when it differs from next edge, or is last */
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ed->v1 != (ed + 1)->v1 || ed->v2 != (ed + 1)->v2) final++;
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
|
|
|
final++;
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
|
2012-06-17 09:58:26 +00:00
|
|
|
(*alledge) = medge = MEM_callocN(sizeof(MEdge) * final, "BKE_mesh_make_edges mdge");
|
2012-05-06 15:15:33 +00:00
|
|
|
(*_totedge) = final;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = totedge, ed = edsort; a > 1; a--, ed++) {
|
2004-07-08 20:38:27 +00:00
|
|
|
/* edge is unique when it differs from next edge, or is last */
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ed->v1 != (ed + 1)->v1 || ed->v2 != (ed + 1)->v2) {
|
|
|
|
medge->v1 = ed->v1;
|
|
|
|
medge->v2 = ed->v2;
|
|
|
|
if (old == 0 || ed->is_draw) medge->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
|
|
if (ed->is_loose) medge->flag |= ME_LOOSEEDGE;
|
2010-03-30 22:43:43 +00:00
|
|
|
|
|
|
|
/* order is swapped so extruding this edge as a surface wont flip face normals
|
|
|
|
* with cyclic curves */
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ed->v1 + 1 != ed->v2) {
|
2011-12-27 03:54:23 +00:00
|
|
|
SWAP(unsigned int, medge->v1, medge->v2);
|
2010-03-30 22:43:43 +00:00
|
|
|
}
|
2004-07-08 20:38:27 +00:00
|
|
|
medge++;
|
|
|
|
}
|
2005-09-22 17:00:58 +00:00
|
|
|
else {
|
|
|
|
/* equal edge, we merge the drawflag */
|
2012-05-06 15:15:33 +00:00
|
|
|
(ed + 1)->is_draw |= ed->is_draw;
|
2005-09-22 17:00:58 +00:00
|
|
|
}
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
|
|
|
/* last edge */
|
2012-05-06 15:15:33 +00:00
|
|
|
medge->v1 = ed->v1;
|
|
|
|
medge->v2 = ed->v2;
|
|
|
|
medge->flag = ME_EDGEDRAW;
|
|
|
|
if (ed->is_loose) medge->flag |= ME_LOOSEEDGE;
|
2005-08-12 21:55:50 +00:00
|
|
|
medge->flag |= ME_EDGERENDER;
|
2004-07-08 20:38:27 +00:00
|
|
|
|
|
|
|
MEM_freeN(edsort);
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2012-08-17 14:43:20 +00:00
|
|
|
/* set edge members of mloops */
|
2012-05-06 15:15:33 +00:00
|
|
|
medge = *alledge;
|
|
|
|
for (a = 0; a < *_totedge; a++, medge++) {
|
2011-04-15 01:19:13 +00:00
|
|
|
BLI_edgehash_insert(hash, medge->v1, medge->v2, SET_INT_IN_POINTER(a));
|
|
|
|
}
|
|
|
|
|
|
|
|
mpoly = allpoly;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < totpoly; a++, mpoly++) {
|
2013-02-28 04:18:01 +00:00
|
|
|
MLoop *ml, *ml_next;
|
|
|
|
int i = mpoly->totloop;
|
|
|
|
|
|
|
|
ml_next = allloop + mpoly->loopstart; /* first loop */
|
|
|
|
ml = &ml_next[i - 1]; /* last loop */
|
|
|
|
|
|
|
|
while (i-- != 0) {
|
|
|
|
ml->e = GET_INT_FROM_POINTER(BLI_edgehash_lookup(hash, ml->v, ml_next->v));
|
|
|
|
ml = ml_next;
|
|
|
|
ml_next++;
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_edgehash_free(hash, NULL);
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_make_edges(Mesh *me, int old)
|
2010-03-05 16:47:52 +00:00
|
|
|
{
|
|
|
|
MEdge *medge;
|
2012-05-06 15:15:33 +00:00
|
|
|
int totedge = 0;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2011-04-15 01:19:13 +00:00
|
|
|
make_edges_mdata(me->mvert, me->mface, me->mloop, me->mpoly, me->totvert, me->totface, me->totloop, me->totpoly, old, &medge, &totedge);
|
2012-05-06 15:15:33 +00:00
|
|
|
if (totedge == 0) {
|
2010-03-05 16:47:52 +00:00
|
|
|
/* flag that mesh has edges */
|
|
|
|
me->medge = medge;
|
|
|
|
me->totedge = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
medge = CustomData_add_layer(&me->edata, CD_MEDGE, CD_ASSIGN, medge, totedge);
|
|
|
|
me->medge = medge;
|
|
|
|
me->totedge = totedge;
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_strip_loose_faces(me);
|
2004-07-08 20:38:27 +00:00
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* We need to keep this for edge creation (for now?), and some old readfile code... */
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_strip_loose_faces(Mesh *me)
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
{
|
2012-03-15 20:10:07 +00:00
|
|
|
MFace *f;
|
|
|
|
int a, b;
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
for (a = b = 0, f = me->mface; a < me->totface; a++, f++) {
|
|
|
|
if (f->v3) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->mface[b], f, sizeof(me->mface[b]));
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
CustomData_copy_data(&me->fdata, &me->fdata, a, b, 1);
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
}
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->fdata, b, a - b);
|
|
|
|
me->totface = b;
|
|
|
|
}
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
}
|
2004-07-08 20:38:27 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Works on both loops and polys! */
|
|
|
|
/* Note: It won't try to guess which loops of an invalid poly to remove!
|
|
|
|
* this is the work of the caller, to mark those loops...
|
|
|
|
* See e.g. BKE_mesh_validate_arrays(). */
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_strip_loose_polysloops(Mesh *me)
|
2011-02-09 15:13:20 +00:00
|
|
|
{
|
2012-03-15 20:10:07 +00:00
|
|
|
MPoly *p;
|
|
|
|
MLoop *l;
|
|
|
|
int a, b;
|
|
|
|
/* New loops idx! */
|
2012-04-22 00:27:38 +00:00
|
|
|
int *new_idx = MEM_mallocN(sizeof(int) * me->totloop, __func__);
|
2012-03-15 20:10:07 +00:00
|
|
|
|
|
|
|
for (a = b = 0, p = me->mpoly; a < me->totpoly; a++, p++) {
|
|
|
|
int invalid = FALSE;
|
|
|
|
int i = p->loopstart;
|
|
|
|
int stop = i + p->totloop;
|
|
|
|
|
|
|
|
if (stop > me->totloop || stop < i) {
|
|
|
|
invalid = TRUE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
l = &me->mloop[i];
|
|
|
|
i = stop - i;
|
|
|
|
/* If one of the poly's loops is invalid, the whole poly is invalid! */
|
|
|
|
for (; i--; l++) {
|
|
|
|
if (l->e == INVALID_LOOP_EDGE_MARKER) {
|
|
|
|
invalid = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p->totloop >= 3 && !invalid) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->mpoly[b], p, sizeof(me->mpoly[b]));
|
|
|
|
CustomData_copy_data(&me->pdata, &me->pdata, a, b, 1);
|
|
|
|
}
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->pdata, b, a - b);
|
|
|
|
me->totpoly = b;
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* And now, get rid of invalid loops. */
|
|
|
|
for (a = b = 0, l = me->mloop; a < me->totloop; a++, l++) {
|
|
|
|
if (l->e != INVALID_LOOP_EDGE_MARKER) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->mloop[b], l, sizeof(me->mloop[b]));
|
|
|
|
CustomData_copy_data(&me->ldata, &me->ldata, a, b, 1);
|
|
|
|
}
|
|
|
|
new_idx[a] = b;
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
else {
|
2012-04-25 06:06:40 +00:00
|
|
|
/* XXX Theoretically, we should be able to not do this, as no remaining poly
|
2012-03-15 20:10:07 +00:00
|
|
|
* should use any stripped loop. But for security's sake... */
|
|
|
|
new_idx[a] = -a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->ldata, b, a - b);
|
|
|
|
me->totloop = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And now, update polys' start loop index. */
|
|
|
|
/* Note: At this point, there should never be any poly using a striped loop! */
|
|
|
|
for (a = 0, p = me->mpoly; a < me->totpoly; a++, p++) {
|
|
|
|
p->loopstart = new_idx[p->loopstart];
|
|
|
|
}
|
2012-04-22 00:27:38 +00:00
|
|
|
|
|
|
|
MEM_freeN(new_idx);
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_strip_loose_edges(Mesh *me)
|
2012-03-15 20:10:07 +00:00
|
|
|
{
|
|
|
|
MEdge *e;
|
|
|
|
MLoop *l;
|
|
|
|
int a, b;
|
2012-04-26 15:38:16 +00:00
|
|
|
unsigned int *new_idx = MEM_mallocN(sizeof(int) * me->totedge, __func__);
|
2012-03-15 20:10:07 +00:00
|
|
|
|
|
|
|
for (a = b = 0, e = me->medge; a < me->totedge; a++, e++) {
|
|
|
|
if (e->v1 != e->v2) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->medge[b], e, sizeof(me->medge[b]));
|
2011-02-09 15:13:20 +00:00
|
|
|
CustomData_copy_data(&me->edata, &me->edata, a, b, 1);
|
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
new_idx[a] = b;
|
2011-02-09 15:13:20 +00:00
|
|
|
b++;
|
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
else {
|
|
|
|
new_idx[a] = INVALID_LOOP_EDGE_MARKER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->edata, b, a - b);
|
|
|
|
me->totedge = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And now, update loops' edge indices. */
|
|
|
|
/* XXX We hope no loop was pointing to a striped edge!
|
|
|
|
* Else, its e will be set to INVALID_LOOP_EDGE_MARKER :/ */
|
|
|
|
for (a = 0, l = me->mloop; a < me->totloop; a++, l++) {
|
|
|
|
l->e = new_idx[l->e];
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2012-04-26 15:20:26 +00:00
|
|
|
|
|
|
|
MEM_freeN(new_idx);
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_from_metaball(ListBase *lb, Mesh *me)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
DispList *dl;
|
|
|
|
MVert *mvert;
|
2012-03-20 14:23:22 +00:00
|
|
|
MLoop *mloop, *allloop;
|
|
|
|
MPoly *mpoly;
|
2002-10-12 11:37:38 +00:00
|
|
|
float *nors, *verts;
|
|
|
|
int a, *index;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
dl = lb->first;
|
|
|
|
if (dl == NULL) return;
|
|
|
|
|
|
|
|
if (dl->type == DL_INDEX4) {
|
|
|
|
mvert = CustomData_add_layer(&me->vdata, CD_MVERT, CD_CALLOC, NULL, dl->nr);
|
|
|
|
allloop = mloop = CustomData_add_layer(&me->ldata, CD_MLOOP, CD_CALLOC, NULL, dl->parts * 4);
|
|
|
|
mpoly = CustomData_add_layer(&me->pdata, CD_MPOLY, CD_CALLOC, NULL, dl->parts);
|
|
|
|
me->mvert = mvert;
|
|
|
|
me->mloop = mloop;
|
|
|
|
me->mpoly = mpoly;
|
|
|
|
me->totvert = dl->nr;
|
|
|
|
me->totpoly = dl->parts;
|
|
|
|
|
|
|
|
a = dl->nr;
|
|
|
|
nors = dl->nors;
|
|
|
|
verts = dl->verts;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2011-11-20 16:21:13 +00:00
|
|
|
copy_v3_v3(mvert->co, verts);
|
2011-03-28 02:34:55 +00:00
|
|
|
normal_float_to_short_v3(mvert->no, nors);
|
2002-10-12 11:37:38 +00:00
|
|
|
mvert++;
|
2012-05-06 15:15:33 +00:00
|
|
|
nors += 3;
|
|
|
|
verts += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
a = dl->parts;
|
|
|
|
index = dl->index;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2012-05-06 15:15:33 +00:00
|
|
|
int count = index[2] != index[3] ? 4 : 3;
|
2006-11-23 00:26:39 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
mloop[0].v = index[0];
|
|
|
|
mloop[1].v = index[1];
|
|
|
|
mloop[2].v = index[2];
|
2012-03-20 14:23:22 +00:00
|
|
|
if (count == 4)
|
2012-05-06 15:15:33 +00:00
|
|
|
mloop[3].v = index[3];
|
2006-11-23 00:26:39 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
mpoly->totloop = count;
|
|
|
|
mpoly->loopstart = (int)(mloop - allloop);
|
|
|
|
mpoly->flag = ME_SMOOTH;
|
2012-03-20 14:23:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
mpoly++;
|
2012-05-06 15:15:33 +00:00
|
|
|
mloop += count;
|
|
|
|
me->totloop += count;
|
|
|
|
index += 4;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
BKE_mesh_update_customdata_pointers(me, true);
|
2012-01-24 00:52:26 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_calc_normals(me->mvert, me->totvert, me->mloop, me->mpoly, me->totloop, me->totpoly, NULL);
|
2011-11-13 15:13:59 +00:00
|
|
|
|
2013-03-16 01:19:03 +00:00
|
|
|
BKE_mesh_calc_edges(me, true, false);
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2010-03-05 16:47:52 +00:00
|
|
|
/* Initialize mverts, medges and, faces for converting nurbs to mesh and derived mesh */
|
|
|
|
/* return non-zero on error */
|
2012-05-05 21:28:12 +00:00
|
|
|
int BKE_mesh_nurbs_to_mdata(Object *ob, MVert **allvert, int *totvert,
|
|
|
|
MEdge **alledge, int *totedge, MLoop **allloop, MPoly **allpoly,
|
|
|
|
int *totloop, int *totpoly)
|
2010-03-08 13:49:13 +00:00
|
|
|
{
|
2012-07-14 17:30:49 +00:00
|
|
|
return BKE_mesh_nurbs_displist_to_mdata(ob, &ob->disp,
|
2012-05-05 21:28:12 +00:00
|
|
|
allvert, totvert,
|
|
|
|
alledge, totedge,
|
2013-01-10 17:37:17 +00:00
|
|
|
allloop, allpoly, NULL,
|
|
|
|
totloop, totpoly);
|
2010-03-08 13:49:13 +00:00
|
|
|
}
|
|
|
|
|
2012-02-03 04:58:55 +00:00
|
|
|
/* BMESH: this doesn't calculate all edges from polygons,
|
|
|
|
* only free standing edges are calculated */
|
|
|
|
|
2010-03-08 13:49:13 +00:00
|
|
|
/* Initialize mverts, medges and, faces for converting nurbs to mesh and derived mesh */
|
2012-07-31 13:43:26 +00:00
|
|
|
/* use specified dispbase */
|
2012-07-14 17:30:49 +00:00
|
|
|
int BKE_mesh_nurbs_displist_to_mdata(Object *ob, ListBase *dispbase,
|
2012-05-05 21:28:12 +00:00
|
|
|
MVert **allvert, int *_totvert,
|
|
|
|
MEdge **alledge, int *_totedge,
|
|
|
|
MLoop **allloop, MPoly **allpoly,
|
2013-01-10 17:37:17 +00:00
|
|
|
MLoopUV **alluv,
|
|
|
|
int *_totloop, int *_totpoly)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
DispList *dl;
|
|
|
|
Curve *cu;
|
|
|
|
MVert *mvert;
|
2011-04-15 01:19:13 +00:00
|
|
|
MPoly *mpoly;
|
|
|
|
MLoop *mloop;
|
2013-01-10 17:37:17 +00:00
|
|
|
MLoopUV *mloopuv = NULL;
|
2012-02-03 04:58:55 +00:00
|
|
|
MEdge *medge;
|
2002-10-12 11:37:38 +00:00
|
|
|
float *data;
|
2012-05-06 15:15:33 +00:00
|
|
|
int a, b, ofs, vertcount, startvert, totvert = 0, totedge = 0, totloop = 0, totvlak = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
int p1, p2, p3, p4, *index;
|
2012-05-06 15:15:33 +00:00
|
|
|
int conv_polys = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
cu = ob->data;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
conv_polys |= cu->flag & CU_3D; /* 2d polys are filled with DL_INDEX3 displists */
|
|
|
|
conv_polys |= ob->type == OB_SURF; /* surf polys are never filled */
|
2010-11-06 06:22:25 +00:00
|
|
|
|
2003-04-26 13:07:59 +00:00
|
|
|
/* count */
|
2012-05-06 15:15:33 +00:00
|
|
|
dl = dispbase->first;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (dl) {
|
2012-05-06 15:15:33 +00:00
|
|
|
if (dl->type == DL_SEGM) {
|
|
|
|
totvert += dl->parts * dl->nr;
|
|
|
|
totedge += dl->parts * (dl->nr - 1);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (dl->type == DL_POLY) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (conv_polys) {
|
2012-05-06 15:15:33 +00:00
|
|
|
totvert += dl->parts * dl->nr;
|
|
|
|
totedge += dl->parts * dl->nr;
|
2010-11-06 06:22:25 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (dl->type == DL_SURF) {
|
2012-02-03 04:58:55 +00:00
|
|
|
int tot;
|
2012-05-06 15:15:33 +00:00
|
|
|
totvert += dl->parts * dl->nr;
|
|
|
|
tot = (dl->parts - 1 + ((dl->flag & DL_CYCL_V) == 2)) * (dl->nr - 1 + (dl->flag & DL_CYCL_U));
|
2012-02-03 04:58:55 +00:00
|
|
|
totvlak += tot;
|
|
|
|
totloop += tot * 4;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (dl->type == DL_INDEX3) {
|
2012-02-03 04:58:55 +00:00
|
|
|
int tot;
|
2012-05-06 15:15:33 +00:00
|
|
|
totvert += dl->nr;
|
2012-02-03 04:58:55 +00:00
|
|
|
tot = dl->parts;
|
2012-05-06 15:15:33 +00:00
|
|
|
totvlak += tot;
|
2012-02-03 04:58:55 +00:00
|
|
|
totloop += tot * 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (totvert == 0) {
|
2007-05-28 16:49:48 +00:00
|
|
|
/* error("can't convert"); */
|
|
|
|
/* Make Sure you check ob->data is a curve */
|
2010-03-05 16:47:52 +00:00
|
|
|
return -1;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-02-03 04:58:55 +00:00
|
|
|
*allvert = mvert = MEM_callocN(sizeof(MVert) * totvert, "nurbs_init mvert");
|
|
|
|
*alledge = medge = MEM_callocN(sizeof(MEdge) * totedge, "nurbs_init medge");
|
|
|
|
*allloop = mloop = MEM_callocN(sizeof(MLoop) * totvlak * 4, "nurbs_init mloop"); // totloop
|
|
|
|
*allpoly = mpoly = MEM_callocN(sizeof(MPoly) * totvlak, "nurbs_init mloop");
|
2013-01-10 17:37:17 +00:00
|
|
|
|
|
|
|
if (alluv)
|
|
|
|
*alluv = mloopuv = MEM_callocN(sizeof(MLoopUV) * totvlak * 4, "nurbs_init mloopuv");
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2003-04-26 13:07:59 +00:00
|
|
|
/* verts and faces */
|
2012-05-06 15:15:33 +00:00
|
|
|
vertcount = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
dl = dispbase->first;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (dl) {
|
2012-05-06 15:15:33 +00:00
|
|
|
int smooth = dl->rt & CU_SMOOTH ? 1 : 0;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (dl->type == DL_SEGM) {
|
|
|
|
startvert = vertcount;
|
|
|
|
a = dl->parts * dl->nr;
|
|
|
|
data = dl->verts;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2011-11-20 16:21:13 +00:00
|
|
|
copy_v3_v3(mvert->co, data);
|
2012-05-06 15:15:33 +00:00
|
|
|
data += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
vertcount++;
|
|
|
|
mvert++;
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < dl->parts; a++) {
|
|
|
|
ofs = a * dl->nr;
|
|
|
|
for (b = 1; b < dl->nr; b++) {
|
|
|
|
medge->v1 = startvert + ofs + b - 1;
|
|
|
|
medge->v2 = startvert + ofs + b;
|
2012-04-21 13:08:26 +00:00
|
|
|
medge->flag = ME_LOOSEEDGE | ME_EDGERENDER | ME_EDGEDRAW;
|
2012-02-03 04:58:55 +00:00
|
|
|
|
|
|
|
medge++;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (dl->type == DL_POLY) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (conv_polys) {
|
2012-05-06 15:15:33 +00:00
|
|
|
startvert = vertcount;
|
|
|
|
a = dl->parts * dl->nr;
|
|
|
|
data = dl->verts;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2011-11-20 16:21:13 +00:00
|
|
|
copy_v3_v3(mvert->co, data);
|
2012-05-06 15:15:33 +00:00
|
|
|
data += 3;
|
2010-11-06 06:22:25 +00:00
|
|
|
vertcount++;
|
|
|
|
mvert++;
|
|
|
|
}
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < dl->parts; a++) {
|
|
|
|
ofs = a * dl->nr;
|
|
|
|
for (b = 0; b < dl->nr; b++) {
|
|
|
|
medge->v1 = startvert + ofs + b;
|
|
|
|
if (b == dl->nr - 1) medge->v2 = startvert + ofs;
|
|
|
|
else medge->v2 = startvert + ofs + b + 1;
|
2012-04-21 13:08:26 +00:00
|
|
|
medge->flag = ME_LOOSEEDGE | ME_EDGERENDER | ME_EDGEDRAW;
|
2012-02-03 04:58:55 +00:00
|
|
|
medge++;
|
2010-11-06 06:22:25 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (dl->type == DL_INDEX3) {
|
|
|
|
startvert = vertcount;
|
|
|
|
a = dl->nr;
|
|
|
|
data = dl->verts;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2011-11-20 16:21:13 +00:00
|
|
|
copy_v3_v3(mvert->co, data);
|
2012-05-06 15:15:33 +00:00
|
|
|
data += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
vertcount++;
|
|
|
|
mvert++;
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
a = dl->parts;
|
|
|
|
index = dl->index;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2012-05-06 15:15:33 +00:00
|
|
|
mloop[0].v = startvert + index[0];
|
|
|
|
mloop[1].v = startvert + index[2];
|
|
|
|
mloop[2].v = startvert + index[1];
|
2012-02-03 04:58:55 +00:00
|
|
|
mpoly->loopstart = (int)(mloop - (*allloop));
|
|
|
|
mpoly->totloop = 3;
|
|
|
|
mpoly->mat_nr = dl->col;
|
|
|
|
|
2013-01-10 17:37:17 +00:00
|
|
|
if (mloopuv) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < 3; i++, mloopuv++) {
|
2013-01-12 14:28:23 +00:00
|
|
|
mloopuv->uv[0] = (mloop[i].v - startvert) / (float)(dl->nr - 1);
|
2013-01-10 17:37:17 +00:00
|
|
|
mloopuv->uv[1] = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (smooth) mpoly->flag |= ME_SMOOTH;
|
2012-02-03 04:58:55 +00:00
|
|
|
mpoly++;
|
2012-05-06 15:15:33 +00:00
|
|
|
mloop += 3;
|
|
|
|
index += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (dl->type == DL_SURF) {
|
|
|
|
startvert = vertcount;
|
|
|
|
a = dl->parts * dl->nr;
|
|
|
|
data = dl->verts;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (a--) {
|
2011-11-20 16:21:13 +00:00
|
|
|
copy_v3_v3(mvert->co, data);
|
2012-05-06 15:15:33 +00:00
|
|
|
data += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
vertcount++;
|
|
|
|
mvert++;
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < dl->parts; a++) {
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if ( (dl->flag & DL_CYCL_V) == 0 && a == dl->parts - 1) break;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (dl->flag & DL_CYCL_U) { /* p2 -> p1 -> */
|
|
|
|
p1 = startvert + dl->nr * a; /* p4 -> p3 -> */
|
|
|
|
p2 = p1 + dl->nr - 1; /* -----> next row */
|
|
|
|
p3 = p1 + dl->nr;
|
|
|
|
p4 = p2 + dl->nr;
|
|
|
|
b = 0;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-05-06 15:15:33 +00:00
|
|
|
p2 = startvert + dl->nr * a;
|
|
|
|
p1 = p2 + 1;
|
|
|
|
p4 = p2 + dl->nr;
|
|
|
|
p3 = p1 + dl->nr;
|
|
|
|
b = 1;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
if ( (dl->flag & DL_CYCL_V) && a == dl->parts - 1) {
|
|
|
|
p3 -= dl->parts * dl->nr;
|
|
|
|
p4 -= dl->parts * dl->nr;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (; b < dl->nr; b++) {
|
|
|
|
mloop[0].v = p1;
|
|
|
|
mloop[1].v = p3;
|
|
|
|
mloop[2].v = p4;
|
|
|
|
mloop[3].v = p2;
|
2012-02-03 04:58:55 +00:00
|
|
|
mpoly->loopstart = (int)(mloop - (*allloop));
|
|
|
|
mpoly->totloop = 4;
|
|
|
|
mpoly->mat_nr = dl->col;
|
|
|
|
|
2013-01-10 17:37:17 +00:00
|
|
|
if (mloopuv) {
|
|
|
|
int orco_sizeu = dl->nr - 1;
|
|
|
|
int orco_sizev = dl->parts - 1;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* exception as handled in convertblender.c too */
|
|
|
|
if (dl->flag & DL_CYCL_U) {
|
|
|
|
orco_sizeu++;
|
|
|
|
if (dl->flag & DL_CYCL_V)
|
|
|
|
orco_sizev++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++, mloopuv++) {
|
|
|
|
/* find uv based on vertex index into grid array */
|
|
|
|
int v = mloop[i].v - startvert;
|
|
|
|
|
2013-01-12 14:28:23 +00:00
|
|
|
mloopuv->uv[0] = (v / dl->nr) / (float)orco_sizev;
|
|
|
|
mloopuv->uv[1] = (v % dl->nr) / (float)orco_sizeu;
|
2013-01-10 17:37:17 +00:00
|
|
|
|
|
|
|
/* cyclic correction */
|
|
|
|
if ((i == 0 || i == 1) && mloopuv->uv[1] == 0.0f)
|
|
|
|
mloopuv->uv[1] = 1.0f;
|
|
|
|
}
|
2012-07-31 13:43:26 +00:00
|
|
|
}
|
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (smooth) mpoly->flag |= ME_SMOOTH;
|
2012-02-03 04:58:55 +00:00
|
|
|
mpoly++;
|
2012-05-06 15:15:33 +00:00
|
|
|
mloop += 4;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
p4 = p3;
|
2002-10-12 11:37:38 +00:00
|
|
|
p3++;
|
2012-05-06 15:15:33 +00:00
|
|
|
p2 = p1;
|
2002-10-12 11:37:38 +00:00
|
|
|
p1++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
*_totpoly = totvlak;
|
|
|
|
*_totloop = totloop;
|
|
|
|
*_totedge = totedge;
|
|
|
|
*_totvert = totvert;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-02-03 04:58:55 +00:00
|
|
|
/* not uded for bmesh */
|
|
|
|
#if 0
|
2011-04-15 01:19:13 +00:00
|
|
|
make_edges_mdata(*allvert, *allface, *allloop, *allpoly, totvert, totvlak, *_totloop, *_totpoly, 0, alledge, _totedge);
|
2010-03-05 16:47:52 +00:00
|
|
|
mfaces_strip_loose(*allface, _totface);
|
2012-02-03 04:58:55 +00:00
|
|
|
#endif
|
2010-03-05 16:47:52 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-07-31 13:43:26 +00:00
|
|
|
|
2010-03-05 16:47:52 +00:00
|
|
|
/* this may fail replacing ob->data, be sure to check ob->type */
|
2013-03-17 19:55:10 +00:00
|
|
|
void BKE_mesh_from_nurbs_displist(Object *ob, ListBase *dispbase, const bool use_orco_uv)
|
2010-03-05 16:47:52 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
Main *bmain = G.main;
|
2010-03-05 16:47:52 +00:00
|
|
|
Object *ob1;
|
2012-05-06 15:15:33 +00:00
|
|
|
DerivedMesh *dm = ob->derivedFinal;
|
2010-03-05 16:47:52 +00:00
|
|
|
Mesh *me;
|
|
|
|
Curve *cu;
|
2012-05-06 15:15:33 +00:00
|
|
|
MVert *allvert = NULL;
|
|
|
|
MEdge *alledge = NULL;
|
2011-04-15 01:19:13 +00:00
|
|
|
MLoop *allloop = NULL;
|
2013-01-10 17:37:17 +00:00
|
|
|
MLoopUV *alluv = NULL;
|
2011-04-15 01:19:13 +00:00
|
|
|
MPoly *allpoly = NULL;
|
2012-02-03 04:58:55 +00:00
|
|
|
int totvert, totedge, totloop, totpoly;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
cu = ob->data;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
|
|
|
if (dm == NULL) {
|
2012-07-14 17:30:49 +00:00
|
|
|
if (BKE_mesh_nurbs_displist_to_mdata(ob, dispbase, &allvert, &totvert,
|
|
|
|
&alledge, &totedge, &allloop,
|
2013-02-02 04:58:03 +00:00
|
|
|
&allpoly, (use_orco_uv) ? &alluv : NULL,
|
2013-01-10 17:37:17 +00:00
|
|
|
&totloop, &totpoly) != 0)
|
2012-05-05 21:28:12 +00:00
|
|
|
{
|
2010-03-05 16:47:52 +00:00
|
|
|
/* Error initializing */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* make mesh */
|
2013-02-05 12:46:15 +00:00
|
|
|
me = BKE_mesh_add(G.main, "Mesh");
|
2012-05-06 15:15:33 +00:00
|
|
|
me->totvert = totvert;
|
|
|
|
me->totedge = totedge;
|
2011-04-15 01:19:13 +00:00
|
|
|
me->totloop = totloop;
|
|
|
|
me->totpoly = totpoly;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
me->mvert = CustomData_add_layer(&me->vdata, CD_MVERT, CD_ASSIGN, allvert, me->totvert);
|
|
|
|
me->medge = CustomData_add_layer(&me->edata, CD_MEDGE, CD_ASSIGN, alledge, me->totedge);
|
|
|
|
me->mloop = CustomData_add_layer(&me->ldata, CD_MLOOP, CD_ASSIGN, allloop, me->totloop);
|
|
|
|
me->mpoly = CustomData_add_layer(&me->pdata, CD_MPOLY, CD_ASSIGN, allpoly, me->totpoly);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2013-01-10 17:37:17 +00:00
|
|
|
if (alluv) {
|
|
|
|
const char *uvname = "Orco";
|
|
|
|
me->mtpoly = CustomData_add_layer_named(&me->pdata, CD_MTEXPOLY, CD_DEFAULT, NULL, me->totpoly, uvname);
|
|
|
|
me->mloopuv = CustomData_add_layer_named(&me->ldata, CD_MLOOPUV, CD_ASSIGN, alluv, me->totloop, uvname);
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_calc_normals(me->mvert, me->totvert, me->mloop, me->mpoly, me->totloop, me->totpoly, NULL);
|
2012-02-03 04:58:55 +00:00
|
|
|
|
2013-03-16 01:19:03 +00:00
|
|
|
BKE_mesh_calc_edges(me, true, false);
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2013-02-05 12:46:15 +00:00
|
|
|
me = BKE_mesh_add(G.main, "Mesh");
|
2011-04-15 05:20:18 +00:00
|
|
|
DM_to_mesh(dm, me, ob);
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
me->totcol = cu->totcol;
|
|
|
|
me->mat = cu->mat;
|
2010-03-14 12:12:21 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_texspace_calc(me);
|
2010-03-14 12:12:21 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
cu->mat = NULL;
|
|
|
|
cu->totcol = 0;
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (ob->data) {
|
2012-05-05 14:03:12 +00:00
|
|
|
BKE_libblock_free(&bmain->curve, ob->data);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
ob->data = me;
|
|
|
|
ob->type = OB_MESH;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2003-04-26 13:07:59 +00:00
|
|
|
/* other users */
|
2012-05-06 15:15:33 +00:00
|
|
|
ob1 = bmain->object.first;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (ob1) {
|
2012-05-06 15:15:33 +00:00
|
|
|
if (ob1->data == cu) {
|
|
|
|
ob1->type = OB_MESH;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
ob1->data = ob->data;
|
2002-10-12 11:37:38 +00:00
|
|
|
id_us_plus((ID *)ob->data);
|
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
ob1 = ob1->id.next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-14 17:30:49 +00:00
|
|
|
void BKE_mesh_from_nurbs(Object *ob)
|
|
|
|
{
|
2013-01-10 17:37:17 +00:00
|
|
|
BKE_mesh_from_nurbs_displist(ob, &ob->disp, false);
|
2012-07-14 17:30:49 +00:00
|
|
|
}
|
|
|
|
|
2009-10-22 23:22:05 +00:00
|
|
|
typedef struct EdgeLink {
|
|
|
|
Link *next, *prev;
|
|
|
|
void *edge;
|
|
|
|
} EdgeLink;
|
|
|
|
|
|
|
|
typedef struct VertLink {
|
|
|
|
Link *next, *prev;
|
2011-12-27 03:54:23 +00:00
|
|
|
unsigned int index;
|
2009-10-22 23:22:05 +00:00
|
|
|
} VertLink;
|
|
|
|
|
2011-12-27 03:54:23 +00:00
|
|
|
static void prependPolyLineVert(ListBase *lb, unsigned int index)
|
2009-10-22 23:22:05 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
VertLink *vl = MEM_callocN(sizeof(VertLink), "VertLink");
|
2009-10-22 23:22:05 +00:00
|
|
|
vl->index = index;
|
|
|
|
BLI_addhead(lb, vl);
|
|
|
|
}
|
|
|
|
|
2011-12-27 03:54:23 +00:00
|
|
|
static void appendPolyLineVert(ListBase *lb, unsigned int index)
|
2009-10-22 23:22:05 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
VertLink *vl = MEM_callocN(sizeof(VertLink), "VertLink");
|
2009-10-22 23:22:05 +00:00
|
|
|
vl->index = index;
|
|
|
|
BLI_addtail(lb, vl);
|
|
|
|
}
|
|
|
|
|
2013-03-15 10:48:48 +00:00
|
|
|
void BKE_mesh_to_curve_nurblist(DerivedMesh *dm, ListBase *nurblist, const int edge_users_test)
|
2009-10-22 23:22:05 +00:00
|
|
|
{
|
2013-03-16 14:33:32 +00:00
|
|
|
MVert *mvert = dm->getVertArray(dm);
|
2012-05-06 15:15:33 +00:00
|
|
|
MEdge *med, *medge = dm->getEdgeArray(dm);
|
2013-03-14 18:35:21 +00:00
|
|
|
MPoly *mp, *mpoly = dm->getPolyArray(dm);
|
|
|
|
MLoop *mloop = dm->getLoopArray(dm);
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2013-03-16 14:33:32 +00:00
|
|
|
int dm_totedge = dm->getNumEdges(dm);
|
|
|
|
int dm_totpoly = dm->getNumPolys(dm);
|
2009-10-22 23:22:05 +00:00
|
|
|
int totedges = 0;
|
2013-03-14 18:35:21 +00:00
|
|
|
int i;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
|
|
|
/* only to detect edge polylines */
|
2013-03-14 18:35:21 +00:00
|
|
|
int *edge_users;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
|
|
|
ListBase edges = {NULL, NULL};
|
|
|
|
|
2013-03-14 18:35:21 +00:00
|
|
|
/* get boundary edges */
|
2013-03-16 14:33:32 +00:00
|
|
|
edge_users = MEM_callocN(sizeof(int) * dm_totedge, __func__);
|
|
|
|
for (i = 0, mp = mpoly; i < dm_totpoly; i++, mp++) {
|
2013-03-14 18:35:21 +00:00
|
|
|
MLoop *ml = &mloop[mp->loopstart];
|
|
|
|
int j;
|
|
|
|
for (j = 0; j < mp->totloop; j++, ml++) {
|
|
|
|
edge_users[ml->e]++;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-14 18:35:21 +00:00
|
|
|
/* create edges from all faces (so as to find edges not in any faces) */
|
2012-05-06 15:15:33 +00:00
|
|
|
med = medge;
|
2013-03-16 14:33:32 +00:00
|
|
|
for (i = 0; i < dm_totedge; i++, med++) {
|
2013-03-15 10:48:48 +00:00
|
|
|
if (edge_users[i] == edge_users_test) {
|
2012-05-06 15:15:33 +00:00
|
|
|
EdgeLink *edl = MEM_callocN(sizeof(EdgeLink), "EdgeLink");
|
|
|
|
edl->edge = med;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
BLI_addtail(&edges, edl); totedges++;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
|
|
|
}
|
2013-03-14 18:35:21 +00:00
|
|
|
MEM_freeN(edge_users);
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (edges.first) {
|
|
|
|
while (edges.first) {
|
2009-10-22 23:22:05 +00:00
|
|
|
/* each iteration find a polyline and add this as a nurbs poly spline */
|
|
|
|
|
|
|
|
ListBase polyline = {NULL, NULL}; /* store a list of VertLink's */
|
|
|
|
int closed = FALSE;
|
2012-05-06 15:15:33 +00:00
|
|
|
int totpoly = 0;
|
|
|
|
MEdge *med_current = ((EdgeLink *)edges.last)->edge;
|
|
|
|
unsigned int startVert = med_current->v1;
|
|
|
|
unsigned int endVert = med_current->v2;
|
|
|
|
int ok = TRUE;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
appendPolyLineVert(&polyline, startVert); totpoly++;
|
|
|
|
appendPolyLineVert(&polyline, endVert); totpoly++;
|
|
|
|
BLI_freelinkN(&edges, edges.last); totedges--;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
while (ok) { /* while connected edges are found... */
|
2009-10-22 23:22:05 +00:00
|
|
|
ok = FALSE;
|
2012-05-06 15:15:33 +00:00
|
|
|
i = totedges;
|
2012-02-23 02:17:50 +00:00
|
|
|
while (i) {
|
2009-10-22 23:22:05 +00:00
|
|
|
EdgeLink *edl;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
i -= 1;
|
|
|
|
edl = BLI_findlink(&edges, i);
|
|
|
|
med = edl->edge;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (med->v1 == endVert) {
|
2009-10-22 23:22:05 +00:00
|
|
|
endVert = med->v2;
|
2012-05-06 15:15:33 +00:00
|
|
|
appendPolyLineVert(&polyline, med->v2); totpoly++;
|
|
|
|
BLI_freelinkN(&edges, edl); totedges--;
|
|
|
|
ok = TRUE;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (med->v2 == endVert) {
|
2009-10-22 23:22:05 +00:00
|
|
|
endVert = med->v1;
|
2012-05-06 15:15:33 +00:00
|
|
|
appendPolyLineVert(&polyline, endVert); totpoly++;
|
|
|
|
BLI_freelinkN(&edges, edl); totedges--;
|
|
|
|
ok = TRUE;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (med->v1 == startVert) {
|
2009-10-22 23:22:05 +00:00
|
|
|
startVert = med->v2;
|
2012-05-06 15:15:33 +00:00
|
|
|
prependPolyLineVert(&polyline, startVert); totpoly++;
|
|
|
|
BLI_freelinkN(&edges, edl); totedges--;
|
|
|
|
ok = TRUE;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
2012-05-06 15:15:33 +00:00
|
|
|
else if (med->v2 == startVert) {
|
2009-10-22 23:22:05 +00:00
|
|
|
startVert = med->v1;
|
2012-05-06 15:15:33 +00:00
|
|
|
prependPolyLineVert(&polyline, startVert); totpoly++;
|
|
|
|
BLI_freelinkN(&edges, edl); totedges--;
|
|
|
|
ok = TRUE;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now we have a polyline, make into a curve */
|
2012-05-06 15:15:33 +00:00
|
|
|
if (startVert == endVert) {
|
2009-10-22 23:22:05 +00:00
|
|
|
BLI_freelinkN(&polyline, polyline.last);
|
|
|
|
totpoly--;
|
|
|
|
closed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --- nurbs --- */
|
|
|
|
{
|
|
|
|
Nurb *nu;
|
|
|
|
BPoint *bp;
|
|
|
|
VertLink *vl;
|
|
|
|
|
|
|
|
/* create new 'nurb' within the curve */
|
|
|
|
nu = (Nurb *)MEM_callocN(sizeof(Nurb), "MeshNurb");
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
nu->pntsu = totpoly;
|
|
|
|
nu->pntsv = 1;
|
|
|
|
nu->orderu = 4;
|
|
|
|
nu->flagu = CU_NURB_ENDPOINT | (closed ? CU_NURB_CYCLIC : 0); /* endpoint */
|
|
|
|
nu->resolu = 12;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
nu->bp = (BPoint *)MEM_callocN(sizeof(BPoint) * totpoly, "bpoints");
|
2009-10-22 23:22:05 +00:00
|
|
|
|
|
|
|
/* add points */
|
2012-05-06 15:15:33 +00:00
|
|
|
vl = polyline.first;
|
|
|
|
for (i = 0, bp = nu->bp; i < totpoly; i++, bp++, vl = (VertLink *)vl->next) {
|
2013-03-16 14:33:32 +00:00
|
|
|
copy_v3_v3(bp->vec, mvert[vl->index].co);
|
2012-05-06 15:15:33 +00:00
|
|
|
bp->f1 = SELECT;
|
2009-10-22 23:22:05 +00:00
|
|
|
bp->radius = bp->weight = 1.0;
|
|
|
|
}
|
|
|
|
BLI_freelistN(&polyline);
|
|
|
|
|
|
|
|
/* add nurb to curve */
|
2013-03-14 18:44:32 +00:00
|
|
|
BLI_addtail(nurblist, nu);
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
|
|
|
/* --- done with nurbs --- */
|
|
|
|
}
|
2013-03-14 18:44:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void BKE_mesh_to_curve(Scene *scene, Object *ob)
|
|
|
|
{
|
|
|
|
/* make new mesh data from the original copy */
|
|
|
|
DerivedMesh *dm = mesh_get_derived_final(scene, ob, CD_MASK_MESH);
|
|
|
|
ListBase nurblist = {NULL, NULL};
|
|
|
|
bool needsFree = false;
|
|
|
|
|
2013-03-15 10:48:48 +00:00
|
|
|
BKE_mesh_to_curve_nurblist(dm, &nurblist, 0);
|
2013-04-08 00:25:44 +00:00
|
|
|
BKE_mesh_to_curve_nurblist(dm, &nurblist, 1);
|
2013-03-14 18:44:32 +00:00
|
|
|
|
|
|
|
if (nurblist.first) {
|
|
|
|
Curve *cu = BKE_curve_add(G.main, ob->id.name + 2, OB_CURVE);
|
|
|
|
cu->flag |= CU_3D;
|
|
|
|
|
|
|
|
cu->nurb = nurblist;
|
2009-10-22 23:22:05 +00:00
|
|
|
|
|
|
|
((Mesh *)ob->data)->id.us--;
|
2012-05-06 15:15:33 +00:00
|
|
|
ob->data = cu;
|
|
|
|
ob->type = OB_CURVE;
|
2010-03-07 05:04:22 +00:00
|
|
|
|
|
|
|
/* curve objects can't contain DM in usual cases, we could free memory */
|
2013-03-14 18:35:21 +00:00
|
|
|
needsFree = true;
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
|
|
|
|
2010-03-07 05:04:22 +00:00
|
|
|
dm->needsFree = needsFree;
|
2009-10-22 23:22:05 +00:00
|
|
|
dm->release(dm);
|
2010-03-07 05:04:22 +00:00
|
|
|
|
|
|
|
if (needsFree) {
|
|
|
|
ob->derivedFinal = NULL;
|
2010-06-23 09:58:02 +00:00
|
|
|
|
|
|
|
/* curve object could have got bounding box only in special cases */
|
2012-02-23 02:17:50 +00:00
|
|
|
if (ob->bb) {
|
2010-06-23 09:58:02 +00:00
|
|
|
MEM_freeN(ob->bb);
|
2012-05-06 15:15:33 +00:00
|
|
|
ob->bb = NULL;
|
2010-06-23 09:58:02 +00:00
|
|
|
}
|
2010-03-07 05:04:22 +00:00
|
|
|
}
|
2009-10-22 23:22:05 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_delete_material_index(Mesh *me, short index)
|
2009-01-04 14:14:06 +00:00
|
|
|
{
|
2004-03-20 23:59:57 +00:00
|
|
|
int i;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->totpoly; i++) {
|
|
|
|
MPoly *mp = &((MPoly *) me->mpoly)[i];
|
|
|
|
if (mp->mat_nr && mp->mat_nr >= index)
|
2011-04-24 08:11:44 +00:00
|
|
|
mp->mat_nr--;
|
2011-04-24 08:06:26 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->totface; i++) {
|
|
|
|
MFace *mf = &((MFace *) me->mface)[i];
|
|
|
|
if (mf->mat_nr && mf->mat_nr >= index)
|
2004-03-20 23:59:57 +00:00
|
|
|
mf->mat_nr--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_smooth_flag_set(Object *meshOb, int enableSmooth)
|
2009-01-04 14:14:06 +00:00
|
|
|
{
|
2005-07-14 15:48:01 +00:00
|
|
|
Mesh *me = meshOb->data;
|
2004-03-20 23:59:57 +00:00
|
|
|
int i;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->totpoly; i++) {
|
|
|
|
MPoly *mp = &((MPoly *) me->mpoly)[i];
|
2011-04-24 08:06:26 +00:00
|
|
|
|
|
|
|
if (enableSmooth) {
|
|
|
|
mp->flag |= ME_SMOOTH;
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-04-24 08:06:26 +00:00
|
|
|
mp->flag &= ~ME_SMOOTH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < me->totface; i++) {
|
|
|
|
MFace *mf = &((MFace *) me->mface)[i];
|
2004-03-20 23:59:57 +00:00
|
|
|
|
|
|
|
if (enableSmooth) {
|
|
|
|
mf->flag |= ME_SMOOTH;
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2004-03-20 23:59:57 +00:00
|
|
|
mf->flag &= ~ME_SMOOTH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-07-17 21:24:43 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_calc_normals_mapping(MVert *mverts, int numVerts,
|
2012-05-06 15:15:33 +00:00
|
|
|
MLoop *mloop, MPoly *mpolys, int numLoops, int numPolys, float (*polyNors_r)[3],
|
|
|
|
MFace *mfaces, int numFaces, int *origIndexFace, float (*faceNors_r)[3])
|
2011-12-06 22:55:41 +00:00
|
|
|
{
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_calc_normals_mapping_ex(mverts, numVerts, mloop, mpolys,
|
|
|
|
numLoops, numPolys, polyNors_r, mfaces, numFaces,
|
|
|
|
origIndexFace, faceNors_r, FALSE);
|
2011-12-06 22:55:41 +00:00
|
|
|
}
|
2012-01-06 00:08:37 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_calc_normals_mapping_ex(MVert *mverts, int numVerts,
|
2012-05-06 15:15:33 +00:00
|
|
|
MLoop *mloop, MPoly *mpolys,
|
|
|
|
int numLoops, int numPolys, float (*polyNors_r)[3],
|
|
|
|
MFace *mfaces, int numFaces, int *origIndexFace, float (*faceNors_r)[3],
|
2013-03-19 23:17:44 +00:00
|
|
|
const bool only_face_normals)
|
2011-04-15 01:19:13 +00:00
|
|
|
{
|
|
|
|
float (*pnors)[3] = polyNors_r, (*fnors)[3] = faceNors_r;
|
2012-01-06 00:08:37 +00:00
|
|
|
int i;
|
2011-04-15 01:19:13 +00:00
|
|
|
MFace *mf;
|
|
|
|
MPoly *mp;
|
2011-09-30 05:27:44 +00:00
|
|
|
|
|
|
|
if (numPolys == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-08-18 03:55:19 +00:00
|
|
|
/* if we are not calculating verts and no verts were passes then we have nothing to do */
|
2011-12-06 22:55:41 +00:00
|
|
|
if ((only_face_normals == TRUE) && (polyNors_r == NULL) && (faceNors_r == NULL)) {
|
|
|
|
printf("%s: called with nothing to do\n", __func__);
|
2011-09-30 05:27:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-12-06 22:55:41 +00:00
|
|
|
if (!pnors) pnors = MEM_callocN(sizeof(float) * 3 * numPolys, "poly_nors mesh.c");
|
|
|
|
/* if (!fnors) fnors = MEM_callocN(sizeof(float) * 3 * numFaces, "face nors mesh.c"); */ /* NO NEED TO ALLOC YET */
|
|
|
|
|
|
|
|
|
|
|
|
if (only_face_normals == FALSE) {
|
|
|
|
/* vertex normals are optional, they require some extra calculations,
|
|
|
|
* so make them optional */
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_calc_normals(mverts, numVerts, mloop, mpolys, numLoops, numPolys, pnors);
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
2011-12-06 22:55:41 +00:00
|
|
|
else {
|
|
|
|
/* only calc poly normals */
|
|
|
|
mp = mpolys;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numPolys; i++, mp++) {
|
2012-09-25 00:20:42 +00:00
|
|
|
BKE_mesh_calc_poly_normal(mp, mloop + mp->loopstart, mverts, pnors[i]);
|
2011-12-06 22:55:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (origIndexFace &&
|
2012-05-27 19:40:36 +00:00
|
|
|
/* fnors == faceNors_r */ /* NO NEED TO ALLOC YET */
|
2012-05-06 15:15:33 +00:00
|
|
|
fnors != NULL &&
|
|
|
|
numFaces)
|
2011-12-06 22:55:41 +00:00
|
|
|
{
|
2011-04-15 01:19:13 +00:00
|
|
|
mf = mfaces;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numFaces; i++, mf++, origIndexFace++) {
|
2011-04-15 02:18:24 +00:00
|
|
|
if (*origIndexFace < numPolys) {
|
2011-11-20 16:21:13 +00:00
|
|
|
copy_v3_v3(fnors[i], pnors[*origIndexFace]);
|
2012-02-25 16:04:03 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* eek, we're not corresponding to polys */
|
2012-05-05 21:28:12 +00:00
|
|
|
printf("error in BKE_mesh_calc_normals; tessellation face indices are incorrect. normals may look bad.\n");
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-09-30 05:27:44 +00:00
|
|
|
|
2011-12-06 22:55:41 +00:00
|
|
|
if (pnors != polyNors_r) MEM_freeN(pnors);
|
|
|
|
/* if (fnors != faceNors_r) MEM_freeN(fnors); */ /* NO NEED TO ALLOC YET */
|
|
|
|
|
2011-04-15 01:19:13 +00:00
|
|
|
fnors = pnors = NULL;
|
|
|
|
|
2004-03-20 23:59:57 +00:00
|
|
|
}
|
2005-07-17 21:24:43 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_calc_normals(MVert *mverts, int numVerts, MLoop *mloop, MPoly *mpolys,
|
|
|
|
int UNUSED(numLoops), int numPolys, float (*polyNors_r)[3])
|
2012-01-06 00:08:37 +00:00
|
|
|
{
|
|
|
|
float (*pnors)[3] = polyNors_r;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
float (*tnorms)[3], (*edgevecbuf)[3] = NULL;
|
2012-01-06 00:08:37 +00:00
|
|
|
float **vertcos = NULL, **vertnos = NULL;
|
|
|
|
BLI_array_declare(vertcos);
|
|
|
|
BLI_array_declare(vertnos);
|
|
|
|
BLI_array_declare(edgevecbuf);
|
|
|
|
|
|
|
|
int i, j;
|
|
|
|
MPoly *mp;
|
|
|
|
MLoop *ml;
|
|
|
|
|
|
|
|
if (!pnors) pnors = MEM_callocN(sizeof(float) * 3 * numPolys, "poly_nors mesh.c");
|
|
|
|
|
2012-08-18 03:55:19 +00:00
|
|
|
/* first go through and calculate normals for all the polys */
|
2012-05-06 15:15:33 +00:00
|
|
|
tnorms = MEM_callocN(sizeof(float) * 3 * numVerts, "tnorms mesh.c");
|
2012-01-06 00:08:37 +00:00
|
|
|
|
|
|
|
mp = mpolys;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numPolys; i++, mp++) {
|
2012-09-25 00:20:42 +00:00
|
|
|
BKE_mesh_calc_poly_normal(mp, mloop + mp->loopstart, mverts, pnors[i]);
|
2012-01-06 00:08:37 +00:00
|
|
|
ml = mloop + mp->loopstart;
|
|
|
|
|
|
|
|
BLI_array_empty(vertcos);
|
|
|
|
BLI_array_empty(vertnos);
|
2012-04-28 15:14:16 +00:00
|
|
|
BLI_array_grow_items(vertcos, mp->totloop);
|
|
|
|
BLI_array_grow_items(vertnos, mp->totloop);
|
2012-02-05 23:09:07 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < mp->totloop; j++) {
|
2012-01-06 00:08:37 +00:00
|
|
|
int vindex = ml[j].v;
|
2012-02-05 23:09:07 +00:00
|
|
|
vertcos[j] = mverts[vindex].co;
|
|
|
|
vertnos[j] = tnorms[vindex];
|
2012-01-06 00:08:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BLI_array_empty(edgevecbuf);
|
2012-04-28 15:14:16 +00:00
|
|
|
BLI_array_grow_items(edgevecbuf, mp->totloop);
|
2012-01-06 00:08:37 +00:00
|
|
|
|
|
|
|
accumulate_vertex_normals_poly(vertnos, pnors[i], vertcos, edgevecbuf, mp->totloop);
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_array_free(vertcos);
|
|
|
|
BLI_array_free(vertnos);
|
|
|
|
BLI_array_free(edgevecbuf);
|
|
|
|
|
|
|
|
/* following Mesh convention; we use vertex coordinate itself for normal in this case */
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numVerts; i++) {
|
|
|
|
MVert *mv = &mverts[i];
|
|
|
|
float *no = tnorms[i];
|
2012-01-06 00:08:37 +00:00
|
|
|
|
2012-09-17 22:22:06 +00:00
|
|
|
if (UNLIKELY(normalize_v3(no) == 0.0f)) {
|
2012-01-06 00:08:37 +00:00
|
|
|
normalize_v3_v3(no, mv->co);
|
2012-09-17 22:22:06 +00:00
|
|
|
}
|
2012-01-06 00:08:37 +00:00
|
|
|
|
|
|
|
normal_float_to_short_v3(mv->no, no);
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(tnorms);
|
|
|
|
|
|
|
|
if (pnors != polyNors_r) MEM_freeN(pnors);
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_calc_normals_tessface(MVert *mverts, int numVerts, MFace *mfaces, int numFaces, float (*faceNors_r)[3])
|
2005-07-17 21:24:43 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
float (*tnorms)[3] = MEM_callocN(numVerts * sizeof(*tnorms), "tnorms");
|
|
|
|
float (*fnors)[3] = (faceNors_r) ? faceNors_r : MEM_callocN(sizeof(*fnors) * numFaces, "meshnormals");
|
2005-07-17 21:24:43 +00:00
|
|
|
int i;
|
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
2006-11-20 04:28:02 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numFaces; i++) {
|
|
|
|
MFace *mf = &mfaces[i];
|
|
|
|
float *f_no = fnors[i];
|
|
|
|
float *n4 = (mf->v4) ? tnorms[mf->v4] : NULL;
|
|
|
|
float *c4 = (mf->v4) ? mverts[mf->v4].co : NULL;
|
2005-07-17 21:24:43 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (mf->v4)
|
2011-03-20 13:35:35 +00:00
|
|
|
normal_quad_v3(f_no, mverts[mf->v1].co, mverts[mf->v2].co, mverts[mf->v3].co, mverts[mf->v4].co);
|
- added mesh_strip_loose_faces, works in conjunction with make_edges
to get rid of faces with MFace.v3==0
- change all Mesh's to have ->medge now. This is forced by make_edges
on readfile, and in the various exotic important routines, and on
conversion back in python.
- make python NMesh structure always have medges now (needs testing)
- with above two changes it is guarenteed that mf->v3 is never ==0
in main blender code (i.e., all MFace's are actually triangles
or quads) and so I went through and removed all the historic tests
to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
- removed MEdge edcode flag, no longer needed
- added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
2005-08-21 07:19:20 +00:00
|
|
|
else
|
2011-03-20 13:35:35 +00:00
|
|
|
normal_tri_v3(f_no, mverts[mf->v1].co, mverts[mf->v2].co, mverts[mf->v3].co);
|
|
|
|
|
2011-04-04 13:02:12 +00:00
|
|
|
accumulate_vertex_normals(tnorms[mf->v1], tnorms[mf->v2], tnorms[mf->v3], n4,
|
2012-05-06 15:15:33 +00:00
|
|
|
f_no, mverts[mf->v1].co, mverts[mf->v2].co, mverts[mf->v3].co, c4);
|
2005-07-17 21:24:43 +00:00
|
|
|
}
|
2011-03-20 13:35:35 +00:00
|
|
|
|
|
|
|
/* following Mesh convention; we use vertex coordinate itself for normal in this case */
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numVerts; i++) {
|
|
|
|
MVert *mv = &mverts[i];
|
|
|
|
float *no = tnorms[i];
|
2005-07-17 21:24:43 +00:00
|
|
|
|
2012-09-17 22:22:06 +00:00
|
|
|
if (UNLIKELY(normalize_v3(no) == 0.0f)) {
|
2010-08-15 15:14:08 +00:00
|
|
|
normalize_v3_v3(no, mv->co);
|
2012-09-17 22:22:06 +00:00
|
|
|
}
|
2005-07-23 19:03:43 +00:00
|
|
|
|
2010-08-15 15:14:08 +00:00
|
|
|
normal_float_to_short_v3(mv->no, no);
|
2005-07-17 21:24:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(tnorms);
|
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
if (fnors != faceNors_r)
|
2005-07-17 21:24:43 +00:00
|
|
|
MEM_freeN(fnors);
|
|
|
|
}
|
2005-07-22 17:03:50 +00:00
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
static void bm_corners_to_loops_ex(ID *id, CustomData *fdata, CustomData *ldata, CustomData *pdata,
|
|
|
|
MFace *mface, int totloop, int findex, int loopstart, int numTex, int numCol)
|
2011-04-15 01:19:13 +00:00
|
|
|
{
|
|
|
|
MTFace *texface;
|
|
|
|
MTexPoly *texpoly;
|
|
|
|
MCol *mcol;
|
|
|
|
MLoopCol *mloopcol;
|
|
|
|
MLoopUV *mloopuv;
|
|
|
|
MFace *mf;
|
|
|
|
int i;
|
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
mf = mface + findex;
|
2011-05-11 02:14:43 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numTex; i++) {
|
2012-06-04 12:10:38 +00:00
|
|
|
texface = CustomData_get_n(fdata, CD_MTFACE, findex, i);
|
|
|
|
texpoly = CustomData_get_n(pdata, CD_MTEXPOLY, findex, i);
|
|
|
|
|
2012-02-08 09:02:10 +00:00
|
|
|
ME_MTEXFACE_CPY(texpoly, texface);
|
2012-06-04 12:10:38 +00:00
|
|
|
|
|
|
|
mloopuv = CustomData_get_n(ldata, CD_MLOOPUV, loopstart, i);
|
2012-02-05 13:25:42 +00:00
|
|
|
copy_v2_v2(mloopuv->uv, texface->uv[0]); mloopuv++;
|
|
|
|
copy_v2_v2(mloopuv->uv, texface->uv[1]); mloopuv++;
|
|
|
|
copy_v2_v2(mloopuv->uv, texface->uv[2]); mloopuv++;
|
2011-04-15 01:19:13 +00:00
|
|
|
|
|
|
|
if (mf->v4) {
|
2012-02-05 13:25:42 +00:00
|
|
|
copy_v2_v2(mloopuv->uv, texface->uv[3]); mloopuv++;
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numCol; i++) {
|
2012-06-04 12:10:38 +00:00
|
|
|
mloopcol = CustomData_get_n(ldata, CD_MLOOPCOL, loopstart, i);
|
|
|
|
mcol = CustomData_get_n(fdata, CD_MCOL, findex, i);
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2012-03-17 20:39:28 +00:00
|
|
|
MESH_MLOOPCOL_FROM_MCOL(mloopcol, &mcol[0]); mloopcol++;
|
|
|
|
MESH_MLOOPCOL_FROM_MCOL(mloopcol, &mcol[1]); mloopcol++;
|
|
|
|
MESH_MLOOPCOL_FROM_MCOL(mloopcol, &mcol[2]); mloopcol++;
|
2011-04-15 01:19:13 +00:00
|
|
|
if (mf->v4) {
|
2012-03-17 20:39:28 +00:00
|
|
|
MESH_MLOOPCOL_FROM_MCOL(mloopcol, &mcol[3]); mloopcol++;
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
2005-07-17 21:24:43 +00:00
|
|
|
}
|
2012-06-04 12:10:38 +00:00
|
|
|
|
|
|
|
if (CustomData_has_layer(fdata, CD_MDISPS)) {
|
|
|
|
MDisps *ld = CustomData_get(ldata, loopstart, CD_MDISPS);
|
|
|
|
MDisps *fd = CustomData_get(fdata, findex, CD_MDISPS);
|
2011-04-15 01:19:13 +00:00
|
|
|
float (*disps)[3] = fd->disps;
|
2012-10-12 14:35:10 +00:00
|
|
|
int tot = mf->v4 ? 4 : 3;
|
2011-04-15 01:19:13 +00:00
|
|
|
int side, corners;
|
2012-04-13 05:39:27 +00:00
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
if (CustomData_external_test(fdata, CD_MDISPS)) {
|
2012-10-02 00:54:41 +00:00
|
|
|
if (id && fdata->external) {
|
2012-06-04 12:10:38 +00:00
|
|
|
CustomData_external_add(ldata, id, CD_MDISPS,
|
|
|
|
totloop, fdata->external->filename);
|
|
|
|
}
|
2012-04-13 05:39:27 +00:00
|
|
|
}
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2011-04-15 01:19:13 +00:00
|
|
|
corners = multires_mdisp_corners(fd);
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2011-10-09 21:59:29 +00:00
|
|
|
if (corners == 0) {
|
|
|
|
/* Empty MDisp layers appear in at least one of the sintel.blend files.
|
2012-03-03 20:19:11 +00:00
|
|
|
* Not sure why this happens, but it seems fine to just ignore them here.
|
2012-05-27 19:40:36 +00:00
|
|
|
* If (corners == 0) for a non-empty layer though, something went wrong. */
|
2011-10-09 21:59:29 +00:00
|
|
|
BLI_assert(fd->totdisp == 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
side = sqrt(fd->totdisp / corners);
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < tot; i++, disps += side * side, ld++) {
|
|
|
|
ld->totdisp = side * side;
|
2012-05-03 21:35:04 +00:00
|
|
|
ld->level = (int)(logf(side - 1.0f) / (float)M_LN2) + 1;
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2011-10-09 21:59:29 +00:00
|
|
|
if (ld->disps)
|
2012-01-25 20:18:12 +00:00
|
|
|
MEM_freeN(ld->disps);
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
ld->disps = MEM_callocN(sizeof(float) * 3 * side * side, "converted loop mdisps");
|
2011-10-09 21:59:29 +00:00
|
|
|
if (fd->disps) {
|
2012-05-06 15:15:33 +00:00
|
|
|
memcpy(ld->disps, disps, sizeof(float) * 3 * side * side);
|
2011-10-09 21:59:29 +00:00
|
|
|
}
|
2011-08-15 23:38:51 +00:00
|
|
|
}
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-27 01:28:25 +00:00
|
|
|
void BKE_mesh_convert_mfaces_to_mpolys(Mesh *mesh)
|
2012-06-04 12:10:38 +00:00
|
|
|
{
|
|
|
|
BKE_mesh_convert_mfaces_to_mpolys_ex(&mesh->id, &mesh->fdata, &mesh->ldata, &mesh->pdata,
|
|
|
|
mesh->totedge, mesh->totface, mesh->totloop, mesh->totpoly,
|
|
|
|
mesh->medge, mesh->mface,
|
|
|
|
&mesh->totloop, &mesh->totpoly, &mesh->mloop, &mesh->mpoly);
|
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
BKE_mesh_update_customdata_pointers(mesh, true);
|
2012-06-04 12:10:38 +00:00
|
|
|
}
|
|
|
|
|
2012-06-07 09:11:16 +00:00
|
|
|
/* the same as BKE_mesh_convert_mfaces_to_mpolys but oriented to be used in do_versions from readfile.c
|
|
|
|
* the difference is how active/render/clone/stencil indices are handled here
|
|
|
|
*
|
|
|
|
* normally thay're being set from pdata which totally makes sense for meshes which are already
|
|
|
|
* converted to bmesh structures, but when loading older files indices shall be updated in other
|
|
|
|
* way around, so newly added pdata and ldata would have this indices set based on fdata layer
|
|
|
|
*
|
|
|
|
* this is normally only needed when reading older files, in all other cases BKE_mesh_convert_mfaces_to_mpolys
|
|
|
|
* shall be always used
|
|
|
|
*/
|
|
|
|
void BKE_mesh_do_versions_convert_mfaces_to_mpolys(Mesh *mesh)
|
|
|
|
{
|
|
|
|
BKE_mesh_convert_mfaces_to_mpolys_ex(&mesh->id, &mesh->fdata, &mesh->ldata, &mesh->pdata,
|
|
|
|
mesh->totedge, mesh->totface, mesh->totloop, mesh->totpoly,
|
|
|
|
mesh->medge, mesh->mface,
|
|
|
|
&mesh->totloop, &mesh->totpoly, &mesh->mloop, &mesh->mpoly);
|
|
|
|
|
|
|
|
CustomData_bmesh_do_versions_update_active_layers(&mesh->fdata, &mesh->pdata, &mesh->ldata);
|
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
BKE_mesh_update_customdata_pointers(mesh, true);
|
2012-06-07 09:11:16 +00:00
|
|
|
}
|
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
void BKE_mesh_convert_mfaces_to_mpolys_ex(ID *id, CustomData *fdata, CustomData *ldata, CustomData *pdata,
|
|
|
|
int totedge_i, int totface_i, int totloop_i, int totpoly_i,
|
|
|
|
MEdge *medge, MFace *mface,
|
2012-06-06 14:48:39 +00:00
|
|
|
int *totloop_r, int *totpoly_r,
|
|
|
|
MLoop **mloop_r, MPoly **mpoly_r)
|
2011-04-15 01:19:13 +00:00
|
|
|
{
|
|
|
|
MFace *mf;
|
2012-06-04 12:10:38 +00:00
|
|
|
MLoop *ml, *mloop;
|
|
|
|
MPoly *mp, *mpoly;
|
2011-04-15 01:19:13 +00:00
|
|
|
MEdge *me;
|
|
|
|
EdgeHash *eh;
|
|
|
|
int numTex, numCol;
|
2012-06-04 12:10:38 +00:00
|
|
|
int i, j, totloop, totpoly, *polyindex;
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2012-03-09 00:41:09 +00:00
|
|
|
/* just in case some of these layers are filled in (can happen with python created meshes) */
|
2012-06-04 12:10:38 +00:00
|
|
|
CustomData_free(ldata, totloop_i);
|
|
|
|
CustomData_free(pdata, totpoly_i);
|
2012-03-07 21:58:58 +00:00
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
totpoly = totface_i;
|
|
|
|
mpoly = MEM_callocN(sizeof(MPoly) * totpoly, "mpoly converted");
|
|
|
|
CustomData_add_layer(pdata, CD_MPOLY, CD_ASSIGN, mpoly, totpoly);
|
|
|
|
|
|
|
|
numTex = CustomData_number_of_layers(fdata, CD_MTFACE);
|
|
|
|
numCol = CustomData_number_of_layers(fdata, CD_MCOL);
|
2011-04-15 01:19:13 +00:00
|
|
|
|
|
|
|
totloop = 0;
|
2012-06-04 12:10:38 +00:00
|
|
|
mf = mface;
|
|
|
|
for (i = 0; i < totface_i; i++, mf++) {
|
2011-04-15 01:19:13 +00:00
|
|
|
totloop += mf->v4 ? 4 : 3;
|
|
|
|
}
|
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
mloop = MEM_callocN(sizeof(MLoop) * totloop, "mloop converted");
|
|
|
|
|
|
|
|
CustomData_add_layer(ldata, CD_MLOOP, CD_ASSIGN, mloop, totloop);
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
CustomData_to_bmeshpoly(fdata, pdata, ldata, totloop, totpoly);
|
|
|
|
|
|
|
|
if (id) {
|
|
|
|
/* ensure external data is transferred */
|
|
|
|
CustomData_external_read(fdata, id, CD_MASK_MDISPS, totface_i);
|
|
|
|
}
|
2012-04-13 05:39:27 +00:00
|
|
|
|
2011-04-15 01:19:13 +00:00
|
|
|
eh = BLI_edgehash_new();
|
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
/* build edge hash */
|
|
|
|
me = medge;
|
|
|
|
for (i = 0; i < totedge_i; i++, me++) {
|
2011-04-15 01:19:13 +00:00
|
|
|
BLI_edgehash_insert(eh, me->v1, me->v2, SET_INT_IN_POINTER(i));
|
2012-04-23 00:58:17 +00:00
|
|
|
|
|
|
|
/* unrelated but avoid having the FGON flag enabled, so we can reuse it later for something else */
|
|
|
|
me->flag &= ~ME_FGON;
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
|
|
|
|
2012-10-30 19:20:17 +00:00
|
|
|
polyindex = CustomData_get_layer(fdata, CD_ORIGINDEX);
|
2012-06-04 12:10:38 +00:00
|
|
|
|
|
|
|
j = 0; /* current loop index */
|
|
|
|
ml = mloop;
|
|
|
|
mf = mface;
|
|
|
|
mp = mpoly;
|
|
|
|
for (i = 0; i < totface_i; i++, mf++, mp++) {
|
2011-04-15 01:19:13 +00:00
|
|
|
mp->loopstart = j;
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2011-04-15 01:19:13 +00:00
|
|
|
mp->totloop = mf->v4 ? 4 : 3;
|
|
|
|
|
|
|
|
mp->mat_nr = mf->mat_nr;
|
|
|
|
mp->flag = mf->flag;
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2012-05-17 07:59:25 +00:00
|
|
|
# define ML(v1, v2) { \
|
|
|
|
ml->v = mf->v1; ml->e = GET_INT_FROM_POINTER(BLI_edgehash_lookup(eh, mf->v1, mf->v2)); ml++; j++; \
|
|
|
|
} (void)0
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2011-04-15 01:19:13 +00:00
|
|
|
ML(v1, v2);
|
|
|
|
ML(v2, v3);
|
|
|
|
if (mf->v4) {
|
|
|
|
ML(v3, v4);
|
|
|
|
ML(v4, v1);
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-04-15 01:19:13 +00:00
|
|
|
ML(v3, v1);
|
|
|
|
}
|
2012-06-04 12:10:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
# undef ML
|
2011-04-15 01:19:13 +00:00
|
|
|
|
2012-06-04 12:10:38 +00:00
|
|
|
bm_corners_to_loops_ex(id, fdata, ldata, pdata, mface, totloop, i, mp->loopstart, numTex, numCol);
|
|
|
|
|
|
|
|
if (polyindex) {
|
|
|
|
*polyindex = i;
|
|
|
|
polyindex++;
|
|
|
|
}
|
2011-04-15 01:19:13 +00:00
|
|
|
}
|
|
|
|
|
2012-07-04 15:52:07 +00:00
|
|
|
/* note, we don't convert NGons at all, these are not even real ngons,
|
2011-09-09 14:51:30 +00:00
|
|
|
* they have their own UV's, colors etc - its more an editing feature. */
|
2011-04-15 01:19:13 +00:00
|
|
|
|
|
|
|
BLI_edgehash_free(eh, NULL);
|
2012-06-04 12:10:38 +00:00
|
|
|
|
|
|
|
*totpoly_r = totpoly;
|
|
|
|
*totloop_r = totloop;
|
|
|
|
*mpoly_r = mpoly;
|
|
|
|
*mloop_r = mloop;
|
2005-07-17 21:24:43 +00:00
|
|
|
}
|
2005-07-22 17:03:50 +00:00
|
|
|
|
2013-03-26 07:29:01 +00:00
|
|
|
float (*BKE_mesh_vertexCos_get(Mesh *me, int *r_numVerts))[3]
|
2005-07-22 17:03:50 +00:00
|
|
|
{
|
2009-01-04 18:16:34 +00:00
|
|
|
int i, numVerts = me->totvert;
|
2012-05-06 15:15:33 +00:00
|
|
|
float (*cos)[3] = MEM_mallocN(sizeof(*cos) * numVerts, "vertexcos1");
|
2012-01-20 23:03:41 +00:00
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
if (r_numVerts) *r_numVerts = numVerts;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numVerts; i++)
|
2011-11-13 15:13:59 +00:00
|
|
|
copy_v3_v3(cos[i], me->mvert[i].co);
|
2012-01-20 23:03:41 +00:00
|
|
|
|
2009-01-04 18:16:34 +00:00
|
|
|
return cos;
|
2005-07-22 17:03:50 +00:00
|
|
|
}
|
2006-01-10 11:36:57 +00:00
|
|
|
|
2011-10-30 09:19:07 +00:00
|
|
|
|
2012-03-27 04:46:52 +00:00
|
|
|
/* ngon version wip, based on EDBM_uv_vert_map_create */
|
2011-10-30 09:19:07 +00:00
|
|
|
/* this replaces the non bmesh function (in trunk) which takes MTFace's, if we ever need it back we could
|
|
|
|
* but for now this replaces it because its unused. */
|
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
UvVertMap *BKE_mesh_uv_vert_map_create(struct MPoly *mpoly, struct MLoop *mloop, struct MLoopUV *mloopuv,
|
|
|
|
unsigned int totpoly, unsigned int totvert, int selected, float *limit)
|
2006-01-10 11:36:57 +00:00
|
|
|
{
|
|
|
|
UvVertMap *vmap;
|
|
|
|
UvMapVert *buf;
|
2011-10-30 09:19:07 +00:00
|
|
|
MPoly *mp;
|
2006-04-03 20:31:10 +00:00
|
|
|
unsigned int a;
|
2012-05-06 15:15:33 +00:00
|
|
|
int i, totuv, nverts;
|
2006-01-10 11:36:57 +00:00
|
|
|
|
|
|
|
totuv = 0;
|
|
|
|
|
|
|
|
/* generate UvMapVert array */
|
2012-05-06 15:15:33 +00:00
|
|
|
mp = mpoly;
|
|
|
|
for (a = 0; a < totpoly; a++, mp++)
|
2012-02-23 02:17:50 +00:00
|
|
|
if (!selected || (!(mp->flag & ME_HIDE) && (mp->flag & ME_FACE_SEL)))
|
2011-10-30 09:19:07 +00:00
|
|
|
totuv += mp->totloop;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (totuv == 0)
|
2006-01-10 11:36:57 +00:00
|
|
|
return NULL;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
vmap = (UvVertMap *)MEM_callocN(sizeof(*vmap), "UvVertMap");
|
2006-01-10 11:36:57 +00:00
|
|
|
if (!vmap)
|
|
|
|
return NULL;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
vmap->vert = (UvMapVert **)MEM_callocN(sizeof(*vmap->vert) * totvert, "UvMapVert*");
|
|
|
|
buf = vmap->buf = (UvMapVert *)MEM_callocN(sizeof(*vmap->buf) * totuv, "UvMapVert");
|
2006-01-10 11:36:57 +00:00
|
|
|
|
|
|
|
if (!vmap->vert || !vmap->buf) {
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_uv_vert_map_free(vmap);
|
2006-01-10 11:36:57 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
mp = mpoly;
|
|
|
|
for (a = 0; a < totpoly; a++, mp++) {
|
2012-02-23 02:17:50 +00:00
|
|
|
if (!selected || (!(mp->flag & ME_HIDE) && (mp->flag & ME_FACE_SEL))) {
|
2012-05-06 15:15:33 +00:00
|
|
|
nverts = mp->totloop;
|
2006-01-10 11:36:57 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < nverts; i++) {
|
|
|
|
buf->tfindex = i;
|
|
|
|
buf->f = a;
|
2006-01-10 11:36:57 +00:00
|
|
|
buf->separate = 0;
|
2012-05-06 15:15:33 +00:00
|
|
|
buf->next = vmap->vert[mloop[mp->loopstart + i].v];
|
|
|
|
vmap->vert[mloop[mp->loopstart + i].v] = buf;
|
2006-01-10 11:36:57 +00:00
|
|
|
buf++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* sort individual uvs for each vert */
|
2012-05-06 15:15:33 +00:00
|
|
|
for (a = 0; a < totvert; a++) {
|
|
|
|
UvMapVert *newvlist = NULL, *vlist = vmap->vert[a];
|
2006-01-10 11:36:57 +00:00
|
|
|
UvMapVert *iterv, *v, *lastv, *next;
|
|
|
|
float *uv, *uv2, uvdiff[2];
|
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
while (vlist) {
|
2012-05-06 15:15:33 +00:00
|
|
|
v = vlist;
|
|
|
|
vlist = vlist->next;
|
|
|
|
v->next = newvlist;
|
|
|
|
newvlist = v;
|
2006-01-10 11:36:57 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
uv = mloopuv[mpoly[v->f].loopstart + v->tfindex].uv;
|
|
|
|
lastv = NULL;
|
|
|
|
iterv = vlist;
|
2006-01-10 11:36:57 +00:00
|
|
|
|
2012-02-23 02:17:50 +00:00
|
|
|
while (iterv) {
|
2012-05-06 15:15:33 +00:00
|
|
|
next = iterv->next;
|
2006-01-10 11:36:57 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
uv2 = mloopuv[mpoly[iterv->f].loopstart + iterv->tfindex].uv;
|
2009-11-10 20:43:45 +00:00
|
|
|
sub_v2_v2v2(uvdiff, uv2, uv);
|
2006-01-10 11:36:57 +00:00
|
|
|
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
if (fabsf(uv[0] - uv2[0]) < limit[0] && fabsf(uv[1] - uv2[1]) < limit[1]) {
|
|
|
|
if (lastv) lastv->next = next;
|
|
|
|
else vlist = next;
|
|
|
|
iterv->next = newvlist;
|
|
|
|
newvlist = iterv;
|
2006-01-10 11:36:57 +00:00
|
|
|
}
|
|
|
|
else
|
2012-05-06 15:15:33 +00:00
|
|
|
lastv = iterv;
|
2006-01-10 11:36:57 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
iterv = next;
|
2006-01-10 11:36:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
newvlist->separate = 1;
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
vmap->vert[a] = newvlist;
|
2006-01-10 11:36:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return vmap;
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
UvMapVert *BKE_mesh_uv_vert_map_get_vert(UvVertMap *vmap, unsigned int v)
|
2006-01-10 11:36:57 +00:00
|
|
|
{
|
|
|
|
return vmap->vert[v];
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_uv_vert_map_free(UvVertMap *vmap)
|
2006-01-10 11:36:57 +00:00
|
|
|
{
|
|
|
|
if (vmap) {
|
|
|
|
if (vmap->vert) MEM_freeN(vmap->vert);
|
|
|
|
if (vmap->buf) MEM_freeN(vmap->buf);
|
|
|
|
MEM_freeN(vmap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-05 06:20:51 +00:00
|
|
|
/* Generates a map where the key is the vertex and the value is a list
|
2012-03-03 20:19:11 +00:00
|
|
|
* of polys that use that vertex as a corner. The lists are allocated
|
|
|
|
* from one memory pool. */
|
2013-03-17 19:55:10 +00:00
|
|
|
void BKE_mesh_vert_poly_map_create(MeshElemMap **map, int **mem,
|
|
|
|
const MPoly *mpoly, const MLoop *mloop,
|
|
|
|
int totvert, int totpoly, int totloop)
|
2012-02-05 06:20:51 +00:00
|
|
|
{
|
2012-03-17 04:41:36 +00:00
|
|
|
int i, j;
|
|
|
|
int *indices;
|
2012-02-05 06:20:51 +00:00
|
|
|
|
2012-03-17 04:41:36 +00:00
|
|
|
(*map) = MEM_callocN(sizeof(MeshElemMap) * totvert, "vert poly map");
|
|
|
|
(*mem) = MEM_mallocN(sizeof(int) * totloop, "vert poly map mem");
|
|
|
|
|
|
|
|
/* Count number of polys for each vertex */
|
|
|
|
for (i = 0; i < totpoly; i++) {
|
|
|
|
const MPoly *p = &mpoly[i];
|
|
|
|
|
|
|
|
for (j = 0; j < p->totloop; j++)
|
|
|
|
(*map)[mloop[p->loopstart + j].v].count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assign indices mem */
|
|
|
|
indices = (*mem);
|
|
|
|
for (i = 0; i < totvert; i++) {
|
|
|
|
(*map)[i].indices = indices;
|
|
|
|
indices += (*map)[i].count;
|
|
|
|
|
|
|
|
/* Reset 'count' for use as index in last loop */
|
|
|
|
(*map)[i].count = 0;
|
|
|
|
}
|
|
|
|
|
2012-02-05 06:20:51 +00:00
|
|
|
/* Find the users */
|
2012-03-17 04:41:36 +00:00
|
|
|
for (i = 0; i < totpoly; i++) {
|
|
|
|
const MPoly *p = &mpoly[i];
|
|
|
|
|
|
|
|
for (j = 0; j < p->totloop; j++) {
|
|
|
|
int v = mloop[p->loopstart + j].v;
|
|
|
|
|
|
|
|
(*map)[v].indices[(*map)[v].count] = i;
|
|
|
|
(*map)[v].count++;
|
2012-02-05 06:20:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-21 22:40:28 +00:00
|
|
|
/* Generates a map where the key is the vertex and the value is a list
|
2012-03-03 20:19:11 +00:00
|
|
|
* of edges that use that vertex as an endpoint. The lists are allocated
|
|
|
|
* from one memory pool. */
|
2013-03-17 19:55:10 +00:00
|
|
|
void BKE_mesh_vert_edge_map_create(MeshElemMap **map, int **mem,
|
|
|
|
const MEdge *medge, int totvert, int totedge)
|
2009-01-21 22:40:28 +00:00
|
|
|
{
|
2012-05-15 16:32:08 +00:00
|
|
|
int i, *indices;
|
|
|
|
|
|
|
|
(*map) = MEM_callocN(sizeof(MeshElemMap) * totvert, "vert-edge map");
|
|
|
|
(*mem) = MEM_mallocN(sizeof(int) * totedge * 2, "vert-edge map mem");
|
2011-04-21 13:11:51 +00:00
|
|
|
|
2012-05-15 16:32:08 +00:00
|
|
|
/* Count number of edges for each vertex */
|
|
|
|
for (i = 0; i < totedge; i++) {
|
|
|
|
(*map)[medge[i].v1].count++;
|
|
|
|
(*map)[medge[i].v2].count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assign indices mem */
|
|
|
|
indices = (*mem);
|
|
|
|
for (i = 0; i < totvert; i++) {
|
|
|
|
(*map)[i].indices = indices;
|
|
|
|
indices += (*map)[i].count;
|
|
|
|
|
|
|
|
/* Reset 'count' for use as index in last loop */
|
|
|
|
(*map)[i].count = 0;
|
|
|
|
}
|
|
|
|
|
2009-01-21 22:40:28 +00:00
|
|
|
/* Find the users */
|
2012-05-15 16:32:08 +00:00
|
|
|
for (i = 0; i < totedge; i++) {
|
|
|
|
const int v[2] = {medge[i].v1, medge[i].v2};
|
|
|
|
|
|
|
|
(*map)[v[0]].indices[(*map)[v[0]].count] = i;
|
|
|
|
(*map)[v[1]].indices[(*map)[v[1]].count] = i;
|
|
|
|
|
|
|
|
(*map)[v[0]].count++;
|
|
|
|
(*map)[v[1]].count++;
|
2009-01-21 22:40:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_loops_to_mface_corners(CustomData *fdata, CustomData *ldata,
|
|
|
|
CustomData *pdata, int lindex[4], int findex,
|
|
|
|
const int polyindex,
|
|
|
|
const int mf_len, /* 3 or 4 */
|
2011-12-28 22:37:09 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
/* cache values to avoid lookups every time */
|
|
|
|
const int numTex, /* CustomData_number_of_layers(pdata, CD_MTEXPOLY) */
|
|
|
|
const int numCol, /* CustomData_number_of_layers(ldata, CD_MLOOPCOL) */
|
|
|
|
const int hasPCol, /* CustomData_has_layer(ldata, CD_PREVIEW_MLOOPCOL) */
|
|
|
|
const int hasOrigSpace /* CustomData_has_layer(ldata, CD_ORIGSPACE_MLOOP) */
|
|
|
|
)
|
2009-08-30 21:30:07 +00:00
|
|
|
{
|
|
|
|
MTFace *texface;
|
|
|
|
MTexPoly *texpoly;
|
|
|
|
MCol *mcol;
|
|
|
|
MLoopCol *mloopcol;
|
|
|
|
MLoopUV *mloopuv;
|
2011-12-28 22:37:09 +00:00
|
|
|
int i, j;
|
2011-03-25 00:32:38 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numTex; i++) {
|
2009-08-30 21:30:07 +00:00
|
|
|
texface = CustomData_get_n(fdata, CD_MTFACE, findex, i);
|
|
|
|
texpoly = CustomData_get_n(pdata, CD_MTEXPOLY, polyindex, i);
|
2012-02-08 09:02:10 +00:00
|
|
|
|
|
|
|
ME_MTEXFACE_CPY(texface, texpoly);
|
2009-08-30 21:30:07 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < mf_len; j++) {
|
2009-08-30 21:30:07 +00:00
|
|
|
mloopuv = CustomData_get_n(ldata, CD_MLOOPUV, lindex[j], i);
|
2012-02-05 13:25:42 +00:00
|
|
|
copy_v2_v2(texface->uv[j], mloopuv->uv);
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numCol; i++) {
|
2009-08-30 21:30:07 +00:00
|
|
|
mcol = CustomData_get_n(fdata, CD_MCOL, findex, i);
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < mf_len; j++) {
|
2009-08-30 21:30:07 +00:00
|
|
|
mloopcol = CustomData_get_n(ldata, CD_MLOOPCOL, lindex[j], i);
|
2012-03-17 20:39:28 +00:00
|
|
|
MESH_MLOOPCOL_TO_MCOL(mloopcol, &mcol[j]);
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
|
|
|
}
|
2009-08-31 15:57:13 +00:00
|
|
|
|
2012-03-22 08:41:50 +00:00
|
|
|
if (hasPCol) {
|
|
|
|
mcol = CustomData_get(fdata, findex, CD_PREVIEW_MCOL);
|
2009-08-31 15:57:13 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < mf_len; j++) {
|
2012-03-22 08:41:50 +00:00
|
|
|
mloopcol = CustomData_get(ldata, lindex[j], CD_PREVIEW_MLOOPCOL);
|
2012-03-17 20:39:28 +00:00
|
|
|
MESH_MLOOPCOL_TO_MCOL(mloopcol, &mcol[j]);
|
2009-08-31 15:57:13 +00:00
|
|
|
}
|
|
|
|
}
|
2012-02-05 11:30:26 +00:00
|
|
|
|
|
|
|
if (hasOrigSpace) {
|
|
|
|
OrigSpaceFace *of = CustomData_get(fdata, findex, CD_ORIGSPACE);
|
|
|
|
OrigSpaceLoop *lof;
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < mf_len; j++) {
|
2012-02-05 11:30:26 +00:00
|
|
|
lof = CustomData_get(ldata, lindex[j], CD_ORIGSPACE_MLOOP);
|
2012-03-18 06:01:33 +00:00
|
|
|
copy_v2_v2(of->uv[j], lof->uv);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
}
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
|
|
|
|
2010-01-28 00:45:30 +00:00
|
|
|
/*
|
2012-03-03 20:19:11 +00:00
|
|
|
* this function recreates a tessellation.
|
|
|
|
* returns number of tessellation faces.
|
2010-01-28 00:45:30 +00:00
|
|
|
*/
|
2012-05-05 21:28:12 +00:00
|
|
|
int BKE_mesh_recalc_tessellation(CustomData *fdata,
|
2012-05-06 15:15:33 +00:00
|
|
|
CustomData *ldata, CustomData *pdata,
|
2012-12-27 06:39:27 +00:00
|
|
|
MVert *mvert, int totface, int totloop,
|
2012-05-06 15:15:33 +00:00
|
|
|
int totpoly,
|
|
|
|
/* when tessellating to recalculate normals after
|
|
|
|
* we can skip copying here */
|
2013-03-17 19:55:10 +00:00
|
|
|
const bool do_face_nor_cpy)
|
2009-08-30 21:30:07 +00:00
|
|
|
{
|
2011-11-18 12:18:44 +00:00
|
|
|
/* use this to avoid locking pthread for _every_ polygon
|
|
|
|
* and calling the fill function */
|
|
|
|
|
|
|
|
#define USE_TESSFACE_SPEEDUP
|
2012-01-25 21:33:37 +00:00
|
|
|
#define USE_TESSFACE_QUADS // NEEDS FURTHER TESTING
|
2012-01-24 20:58:43 +00:00
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
#define TESSFACE_SCANFILL (1 << 0)
|
|
|
|
#define TESSFACE_IS_QUAD (1 << 1)
|
2011-11-18 12:18:44 +00:00
|
|
|
|
2012-12-27 06:39:27 +00:00
|
|
|
const int looptris_tot = poly_to_tri_count(totpoly, totloop);
|
|
|
|
|
2009-08-30 21:30:07 +00:00
|
|
|
MPoly *mp, *mpoly;
|
|
|
|
MLoop *ml, *mloop;
|
2012-12-27 06:39:27 +00:00
|
|
|
MFace *mface, *mf;
|
2012-04-16 06:48:57 +00:00
|
|
|
ScanFillContext sf_ctx;
|
2012-05-13 14:47:53 +00:00
|
|
|
ScanFillVert *sf_vert, *sf_vert_last, *sf_vert_first;
|
|
|
|
ScanFillFace *sf_tri;
|
2012-12-27 06:39:27 +00:00
|
|
|
int *mface_to_poly_map;
|
2011-12-27 08:39:55 +00:00
|
|
|
int lindex[4]; /* only ever use 3 in this case */
|
2012-01-23 13:36:11 +00:00
|
|
|
int poly_index, j, mface_index;
|
2011-12-28 22:37:09 +00:00
|
|
|
|
|
|
|
const int numTex = CustomData_number_of_layers(pdata, CD_MTEXPOLY);
|
|
|
|
const int numCol = CustomData_number_of_layers(ldata, CD_MLOOPCOL);
|
2012-03-22 08:41:50 +00:00
|
|
|
const int hasPCol = CustomData_has_layer(ldata, CD_PREVIEW_MLOOPCOL);
|
2012-02-05 11:30:26 +00:00
|
|
|
const int hasOrigSpace = CustomData_has_layer(ldata, CD_ORIGSPACE_MLOOP);
|
2009-08-30 21:30:07 +00:00
|
|
|
|
|
|
|
mpoly = CustomData_get_layer(pdata, CD_MPOLY);
|
|
|
|
mloop = CustomData_get_layer(ldata, CD_MLOOP);
|
|
|
|
|
2012-01-23 13:36:11 +00:00
|
|
|
/* allocate the length of totfaces, avoid many small reallocs,
|
|
|
|
* if all faces are tri's it will be correct, quads == 2x allocs */
|
2012-12-27 06:39:27 +00:00
|
|
|
/* take care. we are _not_ calloc'ing so be sure to initialize each field */
|
|
|
|
mface_to_poly_map = MEM_mallocN(sizeof(*mface_to_poly_map) * looptris_tot, __func__);
|
|
|
|
mface = MEM_mallocN(sizeof(*mface) * looptris_tot, __func__);
|
2012-01-23 13:36:11 +00:00
|
|
|
|
|
|
|
mface_index = 0;
|
2009-08-30 21:30:07 +00:00
|
|
|
mp = mpoly;
|
2012-01-23 13:36:11 +00:00
|
|
|
for (poly_index = 0; poly_index < totpoly; poly_index++, mp++) {
|
2011-11-18 17:28:19 +00:00
|
|
|
if (mp->totloop < 3) {
|
2011-11-18 12:18:44 +00:00
|
|
|
/* do nothing */
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USE_TESSFACE_SPEEDUP
|
|
|
|
|
|
|
|
#define ML_TO_MF(i1, i2, i3) \
|
2012-01-23 13:36:11 +00:00
|
|
|
mface_to_poly_map[mface_index] = poly_index; \
|
2012-05-06 15:15:33 +00:00
|
|
|
mf = &mface[mface_index]; \
|
2011-11-18 12:18:44 +00:00
|
|
|
/* set loop indices, transformed to vert indices later */ \
|
|
|
|
mf->v1 = mp->loopstart + i1; \
|
|
|
|
mf->v2 = mp->loopstart + i2; \
|
|
|
|
mf->v3 = mp->loopstart + i3; \
|
|
|
|
mf->v4 = 0; \
|
|
|
|
mf->mat_nr = mp->mat_nr; \
|
|
|
|
mf->flag = mp->flag; \
|
2012-12-27 06:39:27 +00:00
|
|
|
mf->edcode = 0; \
|
2012-03-31 03:36:15 +00:00
|
|
|
(void)0
|
2011-11-18 12:18:44 +00:00
|
|
|
|
2012-01-24 20:58:43 +00:00
|
|
|
/* ALMOST IDENTICAL TO DEFINE ABOVE (see EXCEPTION) */
|
|
|
|
#define ML_TO_MF_QUAD() \
|
|
|
|
mface_to_poly_map[mface_index] = poly_index; \
|
2012-05-06 15:15:33 +00:00
|
|
|
mf = &mface[mface_index]; \
|
2012-01-24 20:58:43 +00:00
|
|
|
/* set loop indices, transformed to vert indices later */ \
|
|
|
|
mf->v1 = mp->loopstart + 0; /* EXCEPTION */ \
|
|
|
|
mf->v2 = mp->loopstart + 1; /* EXCEPTION */ \
|
|
|
|
mf->v3 = mp->loopstart + 2; /* EXCEPTION */ \
|
|
|
|
mf->v4 = mp->loopstart + 3; /* EXCEPTION */ \
|
|
|
|
mf->mat_nr = mp->mat_nr; \
|
|
|
|
mf->flag = mp->flag; \
|
2012-12-27 06:39:27 +00:00
|
|
|
mf->edcode = TESSFACE_IS_QUAD; /* EXCEPTION */ \
|
2012-03-31 03:36:15 +00:00
|
|
|
(void)0
|
2012-01-24 20:58:43 +00:00
|
|
|
|
|
|
|
|
2011-11-18 12:18:44 +00:00
|
|
|
else if (mp->totloop == 3) {
|
2012-03-31 03:36:15 +00:00
|
|
|
ML_TO_MF(0, 1, 2);
|
2012-01-23 13:36:11 +00:00
|
|
|
mface_index++;
|
2011-11-18 12:18:44 +00:00
|
|
|
}
|
|
|
|
else if (mp->totloop == 4) {
|
2012-01-24 20:58:43 +00:00
|
|
|
#ifdef USE_TESSFACE_QUADS
|
2012-03-31 03:36:15 +00:00
|
|
|
ML_TO_MF_QUAD();
|
2012-01-24 20:58:43 +00:00
|
|
|
mface_index++;
|
|
|
|
#else
|
2012-03-31 03:36:15 +00:00
|
|
|
ML_TO_MF(0, 1, 2);
|
2012-01-23 13:36:11 +00:00
|
|
|
mface_index++;
|
2012-03-31 03:36:15 +00:00
|
|
|
ML_TO_MF(0, 2, 3);
|
2012-01-23 13:36:11 +00:00
|
|
|
mface_index++;
|
2012-01-24 20:58:43 +00:00
|
|
|
#endif
|
2011-11-18 12:18:44 +00:00
|
|
|
}
|
|
|
|
#endif /* USE_TESSFACE_SPEEDUP */
|
|
|
|
else {
|
2012-01-23 13:36:11 +00:00
|
|
|
int totfilltri;
|
|
|
|
|
2011-02-23 00:01:50 +00:00
|
|
|
ml = mloop + mp->loopstart;
|
2011-04-16 23:58:49 +00:00
|
|
|
|
2012-05-05 00:23:55 +00:00
|
|
|
BLI_scanfill_begin(&sf_ctx);
|
2012-05-13 14:47:53 +00:00
|
|
|
sf_vert_first = NULL;
|
|
|
|
sf_vert_last = NULL;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (j = 0; j < mp->totloop; j++, ml++) {
|
2012-05-13 14:47:53 +00:00
|
|
|
sf_vert = BLI_scanfill_vert_add(&sf_ctx, mvert[ml->v].co);
|
2010-11-10 18:42:53 +00:00
|
|
|
|
2012-05-13 14:47:53 +00:00
|
|
|
sf_vert->keyindex = mp->loopstart + j;
|
2010-11-10 18:42:53 +00:00
|
|
|
|
2012-05-13 14:47:53 +00:00
|
|
|
if (sf_vert_last)
|
|
|
|
BLI_scanfill_edge_add(&sf_ctx, sf_vert_last, sf_vert);
|
2010-11-10 18:42:53 +00:00
|
|
|
|
2012-05-13 14:47:53 +00:00
|
|
|
if (!sf_vert_first)
|
|
|
|
sf_vert_first = sf_vert;
|
|
|
|
sf_vert_last = sf_vert;
|
2010-11-10 18:42:53 +00:00
|
|
|
}
|
2012-05-13 14:47:53 +00:00
|
|
|
BLI_scanfill_edge_add(&sf_ctx, sf_vert_last, sf_vert_first);
|
2010-11-10 18:42:53 +00:00
|
|
|
|
2012-11-26 23:18:04 +00:00
|
|
|
totfilltri = BLI_scanfill_calc(&sf_ctx, 0);
|
2012-12-27 06:39:27 +00:00
|
|
|
BLI_assert(totfilltri <= mp->totloop - 2);
|
2013-02-06 14:02:19 +00:00
|
|
|
(void)totfilltri;
|
2012-01-23 13:36:11 +00:00
|
|
|
|
2012-12-27 06:39:27 +00:00
|
|
|
for (sf_tri = sf_ctx.fillfacebase.first; sf_tri; sf_tri = sf_tri->next, mf++) {
|
|
|
|
mface_to_poly_map[mface_index] = poly_index;
|
|
|
|
mf = &mface[mface_index];
|
2012-01-23 13:36:11 +00:00
|
|
|
|
2012-12-27 06:39:27 +00:00
|
|
|
/* set loop indices, transformed to vert indices later */
|
|
|
|
mf->v1 = sf_tri->v1->keyindex;
|
|
|
|
mf->v2 = sf_tri->v2->keyindex;
|
|
|
|
mf->v3 = sf_tri->v3->keyindex;
|
|
|
|
mf->v4 = 0;
|
2012-01-23 13:36:11 +00:00
|
|
|
|
2012-12-27 06:39:27 +00:00
|
|
|
mf->mat_nr = mp->mat_nr;
|
|
|
|
mf->flag = mp->flag;
|
2011-11-13 15:13:59 +00:00
|
|
|
|
2011-12-29 03:24:28 +00:00
|
|
|
#ifdef USE_TESSFACE_SPEEDUP
|
2012-12-27 06:39:27 +00:00
|
|
|
mf->edcode = TESSFACE_SCANFILL; /* tag for sorting loop indices */
|
2011-12-29 03:24:28 +00:00
|
|
|
#endif
|
|
|
|
|
2012-12-27 06:39:27 +00:00
|
|
|
mface_index++;
|
2010-11-10 18:42:53 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 00:23:55 +00:00
|
|
|
BLI_scanfill_end(&sf_ctx);
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CustomData_free(fdata, totface);
|
2012-01-23 13:36:11 +00:00
|
|
|
totface = mface_index;
|
2012-01-25 23:59:28 +00:00
|
|
|
|
2012-12-27 06:39:27 +00:00
|
|
|
BLI_assert(totface <= looptris_tot);
|
2012-01-25 23:59:28 +00:00
|
|
|
|
2012-02-28 20:55:55 +00:00
|
|
|
/* not essential but without this we store over-alloc'd memory in the CustomData layers */
|
2012-12-27 06:39:27 +00:00
|
|
|
if (LIKELY(looptris_tot != totface)) {
|
2012-01-25 23:59:28 +00:00
|
|
|
mface = MEM_reallocN(mface, sizeof(*mface) * totface);
|
|
|
|
mface_to_poly_map = MEM_reallocN(mface_to_poly_map, sizeof(*mface_to_poly_map) * totface);
|
|
|
|
}
|
|
|
|
|
2011-11-18 12:18:44 +00:00
|
|
|
CustomData_add_layer(fdata, CD_MFACE, CD_ASSIGN, mface, totface);
|
2011-11-23 20:44:04 +00:00
|
|
|
|
2012-10-30 19:20:17 +00:00
|
|
|
/* CD_ORIGINDEX will contain an array of indices from tessfaces to the polygons
|
2012-03-18 07:38:51 +00:00
|
|
|
* they are directly tessellated from */
|
2012-10-30 19:20:17 +00:00
|
|
|
CustomData_add_layer(fdata, CD_ORIGINDEX, CD_ASSIGN, mface_to_poly_map, totface);
|
2009-08-30 21:30:07 +00:00
|
|
|
CustomData_from_bmeshpoly(fdata, pdata, ldata, totface);
|
|
|
|
|
2012-01-19 17:51:52 +00:00
|
|
|
if (do_face_nor_cpy) {
|
|
|
|
/* If polys have a normals layer, copying that to faces can help
|
|
|
|
* avoid the need to recalculate normals later */
|
|
|
|
if (CustomData_has_layer(pdata, CD_NORMAL)) {
|
2012-01-19 22:04:13 +00:00
|
|
|
float (*pnors)[3] = CustomData_get_layer(pdata, CD_NORMAL);
|
|
|
|
float (*fnors)[3] = CustomData_add_layer(fdata, CD_NORMAL, CD_CALLOC, NULL, totface);
|
2012-01-23 13:36:11 +00:00
|
|
|
for (mface_index = 0; mface_index < totface; mface_index++) {
|
|
|
|
copy_v3_v3(fnors[mface_index], pnors[mface_to_poly_map[mface_index]]);
|
2012-01-19 17:51:52 +00:00
|
|
|
}
|
2011-11-13 15:13:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-18 12:18:44 +00:00
|
|
|
mf = mface;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (mface_index = 0; mface_index < totface; mface_index++, mf++) {
|
2011-12-29 03:24:28 +00:00
|
|
|
|
2012-01-24 20:58:43 +00:00
|
|
|
#ifdef USE_TESSFACE_QUADS
|
|
|
|
const int mf_len = mf->edcode & TESSFACE_IS_QUAD ? 4 : 3;
|
|
|
|
#endif
|
|
|
|
|
2011-12-29 03:24:28 +00:00
|
|
|
#ifdef USE_TESSFACE_SPEEDUP
|
|
|
|
/* skip sorting when not using ngons */
|
2012-01-24 20:58:43 +00:00
|
|
|
if (UNLIKELY(mf->edcode & TESSFACE_SCANFILL))
|
2011-12-29 03:24:28 +00:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
/* sort loop indices to ensure winding is correct */
|
2012-08-25 19:43:15 +00:00
|
|
|
if (mf->v1 > mf->v2) SWAP(unsigned int, mf->v1, mf->v2);
|
|
|
|
if (mf->v2 > mf->v3) SWAP(unsigned int, mf->v2, mf->v3);
|
|
|
|
if (mf->v1 > mf->v2) SWAP(unsigned int, mf->v1, mf->v2);
|
2011-12-29 03:24:28 +00:00
|
|
|
|
2012-08-25 19:43:15 +00:00
|
|
|
if (mf->v1 > mf->v2) SWAP(unsigned int, mf->v1, mf->v2);
|
|
|
|
if (mf->v2 > mf->v3) SWAP(unsigned int, mf->v2, mf->v3);
|
|
|
|
if (mf->v1 > mf->v2) SWAP(unsigned int, mf->v1, mf->v2);
|
2012-01-24 20:58:43 +00:00
|
|
|
}
|
2011-12-29 03:24:28 +00:00
|
|
|
|
2012-01-24 20:58:43 +00:00
|
|
|
/* end abusing the edcode */
|
|
|
|
#if defined(USE_TESSFACE_QUADS) || defined(USE_TESSFACE_SPEEDUP)
|
|
|
|
mf->edcode = 0;
|
2011-12-29 03:24:28 +00:00
|
|
|
#endif
|
2012-01-24 20:58:43 +00:00
|
|
|
|
2011-12-29 03:24:28 +00:00
|
|
|
|
2009-08-30 21:30:07 +00:00
|
|
|
lindex[0] = mf->v1;
|
|
|
|
lindex[1] = mf->v2;
|
|
|
|
lindex[2] = mf->v3;
|
2012-01-24 20:58:43 +00:00
|
|
|
#ifdef USE_TESSFACE_QUADS
|
2012-01-25 22:55:12 +00:00
|
|
|
if (mf_len == 4) lindex[3] = mf->v4;
|
2012-01-24 20:58:43 +00:00
|
|
|
#endif
|
2009-08-30 21:30:07 +00:00
|
|
|
|
|
|
|
/*transform loop indices to vert indices*/
|
|
|
|
mf->v1 = mloop[mf->v1].v;
|
|
|
|
mf->v2 = mloop[mf->v2].v;
|
|
|
|
mf->v3 = mloop[mf->v3].v;
|
2012-01-24 20:58:43 +00:00
|
|
|
#ifdef USE_TESSFACE_QUADS
|
|
|
|
if (mf_len == 4) mf->v4 = mloop[mf->v4].v;
|
|
|
|
#endif
|
2009-08-30 21:30:07 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_loops_to_mface_corners(fdata, ldata, pdata,
|
|
|
|
lindex, mface_index, mface_to_poly_map[mface_index],
|
2012-01-24 20:58:43 +00:00
|
|
|
#ifdef USE_TESSFACE_QUADS
|
2012-05-05 21:28:12 +00:00
|
|
|
mf_len,
|
2012-01-24 20:58:43 +00:00
|
|
|
#else
|
2012-05-05 21:28:12 +00:00
|
|
|
3,
|
2012-01-24 20:58:43 +00:00
|
|
|
#endif
|
2012-05-05 21:28:12 +00:00
|
|
|
numTex, numCol, hasPCol, hasOrigSpace);
|
2012-01-24 20:58:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef USE_TESSFACE_QUADS
|
2012-01-25 18:13:58 +00:00
|
|
|
test_index_face(mf, fdata, mface_index, mf_len);
|
2012-01-24 20:58:43 +00:00
|
|
|
#endif
|
|
|
|
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return totface;
|
2011-11-18 16:06:20 +00:00
|
|
|
|
|
|
|
#undef USE_TESSFACE_SPEEDUP
|
|
|
|
|
2009-08-30 21:30:07 +00:00
|
|
|
}
|
2009-09-12 06:47:59 +00:00
|
|
|
|
2011-12-27 08:39:55 +00:00
|
|
|
|
2011-12-28 15:07:00 +00:00
|
|
|
#ifdef USE_BMESH_SAVE_AS_COMPAT
|
2011-12-27 08:39:55 +00:00
|
|
|
|
|
|
|
/*
|
2012-03-02 16:05:54 +00:00
|
|
|
* this function recreates a tessellation.
|
|
|
|
* returns number of tessellation faces.
|
2011-12-27 08:39:55 +00:00
|
|
|
*/
|
2012-05-05 21:28:12 +00:00
|
|
|
int BKE_mesh_mpoly_to_mface(struct CustomData *fdata, struct CustomData *ldata,
|
2012-05-06 15:15:33 +00:00
|
|
|
struct CustomData *pdata, int totface, int UNUSED(totloop), int totpoly)
|
2011-12-27 08:39:55 +00:00
|
|
|
{
|
|
|
|
MLoop *mloop;
|
|
|
|
|
|
|
|
int lindex[4];
|
|
|
|
int i;
|
|
|
|
int k;
|
|
|
|
|
|
|
|
MPoly *mp, *mpoly;
|
|
|
|
MFace *mface = NULL, *mf;
|
|
|
|
BLI_array_declare(mface);
|
|
|
|
|
2011-12-28 22:37:09 +00:00
|
|
|
const int numTex = CustomData_number_of_layers(pdata, CD_MTEXPOLY);
|
|
|
|
const int numCol = CustomData_number_of_layers(ldata, CD_MLOOPCOL);
|
2012-03-22 08:41:50 +00:00
|
|
|
const int hasPCol = CustomData_has_layer(ldata, CD_PREVIEW_MLOOPCOL);
|
2012-02-05 11:30:26 +00:00
|
|
|
const int hasOrigSpace = CustomData_has_layer(ldata, CD_ORIGSPACE_MLOOP);
|
2011-12-28 22:37:09 +00:00
|
|
|
|
2011-12-27 08:39:55 +00:00
|
|
|
mpoly = CustomData_get_layer(pdata, CD_MPOLY);
|
|
|
|
mloop = CustomData_get_layer(ldata, CD_MLOOP);
|
|
|
|
|
|
|
|
mp = mpoly;
|
|
|
|
k = 0;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < totpoly; i++, mp++) {
|
2011-12-27 08:39:55 +00:00
|
|
|
if (ELEM(mp->totloop, 3, 4)) {
|
2012-04-28 15:14:16 +00:00
|
|
|
BLI_array_grow_one(mface);
|
2011-12-27 08:39:55 +00:00
|
|
|
mf = &mface[k];
|
|
|
|
|
|
|
|
mf->mat_nr = mp->mat_nr;
|
|
|
|
mf->flag = mp->flag;
|
|
|
|
|
|
|
|
mf->v1 = mp->loopstart + 0;
|
|
|
|
mf->v2 = mp->loopstart + 1;
|
|
|
|
mf->v3 = mp->loopstart + 2;
|
|
|
|
mf->v4 = (mp->totloop == 4) ? (mp->loopstart + 3) : 0;
|
|
|
|
|
|
|
|
/* abuse edcode for temp storage and clear next loop */
|
|
|
|
mf->edcode = (char)mp->totloop; /* only ever 3 or 4 */
|
|
|
|
|
|
|
|
k++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CustomData_free(fdata, totface);
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
totface = k;
|
2011-12-27 08:39:55 +00:00
|
|
|
|
|
|
|
CustomData_add_layer(fdata, CD_MFACE, CD_ASSIGN, mface, totface);
|
|
|
|
|
|
|
|
CustomData_from_bmeshpoly(fdata, pdata, ldata, totface);
|
|
|
|
|
|
|
|
mp = mpoly;
|
|
|
|
k = 0;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < totpoly; i++, mp++) {
|
2011-12-27 08:39:55 +00:00
|
|
|
if (ELEM(mp->totloop, 3, 4)) {
|
|
|
|
mf = &mface[k];
|
|
|
|
|
|
|
|
if (mf->edcode == 3) {
|
2012-08-17 14:43:20 +00:00
|
|
|
/* sort loop indices to ensure winding is correct */
|
2011-12-27 08:39:55 +00:00
|
|
|
/* NO SORT - looks like we can skip this */
|
|
|
|
|
|
|
|
lindex[0] = mf->v1;
|
|
|
|
lindex[1] = mf->v2;
|
|
|
|
lindex[2] = mf->v3;
|
|
|
|
lindex[3] = 0; /* unused */
|
|
|
|
|
2012-08-17 14:43:20 +00:00
|
|
|
/* transform loop indices to vert indices */
|
2011-12-27 08:39:55 +00:00
|
|
|
mf->v1 = mloop[mf->v1].v;
|
|
|
|
mf->v2 = mloop[mf->v2].v;
|
|
|
|
mf->v3 = mloop[mf->v3].v;
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_loops_to_mface_corners(fdata, ldata, pdata,
|
|
|
|
lindex, k, i, 3,
|
|
|
|
numTex, numCol, hasPCol, hasOrigSpace);
|
2012-01-17 04:54:57 +00:00
|
|
|
test_index_face(mf, fdata, k, 3);
|
2011-12-27 08:39:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-08-17 14:43:20 +00:00
|
|
|
/* sort loop indices to ensure winding is correct */
|
2011-12-27 08:39:55 +00:00
|
|
|
/* NO SORT - looks like we can skip this */
|
|
|
|
|
|
|
|
lindex[0] = mf->v1;
|
|
|
|
lindex[1] = mf->v2;
|
|
|
|
lindex[2] = mf->v3;
|
|
|
|
lindex[3] = mf->v4;
|
|
|
|
|
2012-08-17 14:43:20 +00:00
|
|
|
/* transform loop indices to vert indices */
|
2011-12-27 08:39:55 +00:00
|
|
|
mf->v1 = mloop[mf->v1].v;
|
|
|
|
mf->v2 = mloop[mf->v2].v;
|
|
|
|
mf->v3 = mloop[mf->v3].v;
|
|
|
|
mf->v4 = mloop[mf->v4].v;
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_loops_to_mface_corners(fdata, ldata, pdata,
|
|
|
|
lindex, k, i, 4,
|
|
|
|
numTex, numCol, hasPCol, hasOrigSpace);
|
2012-01-17 04:54:57 +00:00
|
|
|
test_index_face(mf, fdata, k, 4);
|
2011-12-27 08:39:55 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 15:15:33 +00:00
|
|
|
mf->edcode = 0;
|
2011-12-27 08:39:55 +00:00
|
|
|
|
|
|
|
k++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return k;
|
|
|
|
}
|
2011-12-28 15:07:00 +00:00
|
|
|
#endif /* USE_BMESH_SAVE_AS_COMPAT */
|
2011-12-27 08:39:55 +00:00
|
|
|
|
2009-09-12 06:47:59 +00:00
|
|
|
/*
|
|
|
|
* COMPUTE POLY NORMAL
|
|
|
|
*
|
|
|
|
* Computes the normal of a planar
|
|
|
|
* polygon See Graphics Gems for
|
|
|
|
* computing newell normal.
|
|
|
|
*
|
2012-03-03 20:19:11 +00:00
|
|
|
*/
|
2009-09-12 06:47:59 +00:00
|
|
|
static void mesh_calc_ngon_normal(MPoly *mpoly, MLoop *loopstart,
|
2011-11-28 23:50:40 +00:00
|
|
|
MVert *mvert, float normal[3])
|
2009-09-12 06:47:59 +00:00
|
|
|
{
|
2012-05-04 13:13:45 +00:00
|
|
|
const int nverts = mpoly->totloop;
|
|
|
|
float const *v_prev = mvert[loopstart[nverts - 1].v].co;
|
2012-05-12 21:01:26 +00:00
|
|
|
float const *v_curr;
|
2011-05-13 13:17:30 +00:00
|
|
|
int i;
|
2009-09-12 06:47:59 +00:00
|
|
|
|
2012-05-12 21:01:26 +00:00
|
|
|
zero_v3(normal);
|
|
|
|
|
2012-05-04 13:13:45 +00:00
|
|
|
/* Newell's Method */
|
2012-05-12 21:01:26 +00:00
|
|
|
for (i = 0; i < nverts; i++) {
|
|
|
|
v_curr = mvert[loopstart[i].v].co;
|
|
|
|
add_newell_cross_v3_v3v3(normal, v_prev, v_curr);
|
|
|
|
v_prev = v_curr;
|
2009-09-12 06:47:59 +00:00
|
|
|
}
|
|
|
|
|
2012-05-12 21:01:26 +00:00
|
|
|
if (UNLIKELY(normalize_v3(normal) == 0.0f)) {
|
2012-05-04 13:13:45 +00:00
|
|
|
normal[2] = 1.0f; /* other axis set to 0.0 */
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
2009-09-12 06:47:59 +00:00
|
|
|
}
|
|
|
|
|
2012-09-25 00:20:42 +00:00
|
|
|
void BKE_mesh_calc_poly_normal(MPoly *mpoly, MLoop *loopstart,
|
|
|
|
MVert *mvarray, float no[3])
|
2009-09-12 06:47:59 +00:00
|
|
|
{
|
2011-11-28 23:50:40 +00:00
|
|
|
if (mpoly->totloop > 4) {
|
2009-09-12 06:47:59 +00:00
|
|
|
mesh_calc_ngon_normal(mpoly, loopstart, mvarray, no);
|
|
|
|
}
|
2012-02-23 02:17:50 +00:00
|
|
|
else if (mpoly->totloop == 3) {
|
2011-11-28 23:50:40 +00:00
|
|
|
normal_tri_v3(no,
|
|
|
|
mvarray[loopstart[0].v].co,
|
|
|
|
mvarray[loopstart[1].v].co,
|
|
|
|
mvarray[loopstart[2].v].co
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (mpoly->totloop == 4) {
|
|
|
|
normal_quad_v3(no,
|
|
|
|
mvarray[loopstart[0].v].co,
|
|
|
|
mvarray[loopstart[1].v].co,
|
|
|
|
mvarray[loopstart[2].v].co,
|
|
|
|
mvarray[loopstart[3].v].co
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else { /* horrible, two sided face! */
|
|
|
|
no[0] = 0.0;
|
|
|
|
no[1] = 0.0;
|
|
|
|
no[2] = 1.0;
|
|
|
|
}
|
|
|
|
}
|
2012-01-04 14:42:11 +00:00
|
|
|
/* duplicate of function above _but_ takes coords rather then mverts */
|
|
|
|
static void mesh_calc_ngon_normal_coords(MPoly *mpoly, MLoop *loopstart,
|
|
|
|
const float (*vertex_coords)[3], float normal[3])
|
|
|
|
{
|
2012-05-04 13:13:45 +00:00
|
|
|
const int nverts = mpoly->totloop;
|
|
|
|
float const *v_prev = vertex_coords[loopstart[nverts - 1].v];
|
2012-05-12 21:01:26 +00:00
|
|
|
float const *v_curr;
|
2012-01-04 14:42:11 +00:00
|
|
|
int i;
|
|
|
|
|
2012-05-12 21:01:26 +00:00
|
|
|
zero_v3(normal);
|
|
|
|
|
2012-05-04 13:13:45 +00:00
|
|
|
/* Newell's Method */
|
2012-05-12 21:01:26 +00:00
|
|
|
for (i = 0; i < nverts; i++) {
|
|
|
|
v_curr = vertex_coords[loopstart[i].v];
|
|
|
|
add_newell_cross_v3_v3v3(normal, v_prev, v_curr);
|
|
|
|
v_prev = v_curr;
|
2012-01-04 14:42:11 +00:00
|
|
|
}
|
|
|
|
|
2012-05-12 21:01:26 +00:00
|
|
|
if (UNLIKELY(normalize_v3(normal) == 0.0f)) {
|
2012-05-04 13:13:45 +00:00
|
|
|
normal[2] = 1.0f; /* other axis set to 0.0 */
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
2012-01-04 14:42:11 +00:00
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_calc_poly_normal_coords(MPoly *mpoly, MLoop *loopstart,
|
2012-05-06 15:15:33 +00:00
|
|
|
const float (*vertex_coords)[3], float no[3])
|
2012-01-04 14:42:11 +00:00
|
|
|
{
|
|
|
|
if (mpoly->totloop > 4) {
|
|
|
|
mesh_calc_ngon_normal_coords(mpoly, loopstart, vertex_coords, no);
|
|
|
|
}
|
2012-02-23 02:17:50 +00:00
|
|
|
else if (mpoly->totloop == 3) {
|
2012-01-04 14:42:11 +00:00
|
|
|
normal_tri_v3(no,
|
|
|
|
vertex_coords[loopstart[0].v],
|
|
|
|
vertex_coords[loopstart[1].v],
|
|
|
|
vertex_coords[loopstart[2].v]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (mpoly->totloop == 4) {
|
|
|
|
normal_quad_v3(no,
|
|
|
|
vertex_coords[loopstart[0].v],
|
|
|
|
vertex_coords[loopstart[1].v],
|
|
|
|
vertex_coords[loopstart[2].v],
|
|
|
|
vertex_coords[loopstart[3].v]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else { /* horrible, two sided face! */
|
|
|
|
no[0] = 0.0;
|
|
|
|
no[1] = 0.0;
|
|
|
|
no[2] = 1.0;
|
|
|
|
}
|
|
|
|
}
|
2011-11-28 23:50:40 +00:00
|
|
|
|
|
|
|
static void mesh_calc_ngon_center(MPoly *mpoly, MLoop *loopstart,
|
|
|
|
MVert *mvert, float cent[3])
|
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
const float w = 1.0f / (float)mpoly->totloop;
|
2011-11-28 23:50:40 +00:00
|
|
|
int i;
|
2009-09-12 06:47:59 +00:00
|
|
|
|
2011-11-28 23:50:40 +00:00
|
|
|
zero_v3(cent);
|
|
|
|
|
|
|
|
for (i = 0; i < mpoly->totloop; i++) {
|
2011-11-29 02:58:38 +00:00
|
|
|
madd_v3_v3fl(cent, mvert[(loopstart++)->v].co, w);
|
2009-09-12 06:47:59 +00:00
|
|
|
}
|
2011-11-28 23:50:40 +00:00
|
|
|
}
|
2009-09-12 06:47:59 +00:00
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_calc_poly_center(MPoly *mpoly, MLoop *loopstart,
|
2012-05-06 15:15:33 +00:00
|
|
|
MVert *mvarray, float cent[3])
|
2011-11-28 23:50:40 +00:00
|
|
|
{
|
|
|
|
if (mpoly->totloop == 3) {
|
|
|
|
cent_tri_v3(cent,
|
|
|
|
mvarray[loopstart[0].v].co,
|
|
|
|
mvarray[loopstart[1].v].co,
|
|
|
|
mvarray[loopstart[2].v].co
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (mpoly->totloop == 4) {
|
|
|
|
cent_quad_v3(cent,
|
|
|
|
mvarray[loopstart[0].v].co,
|
|
|
|
mvarray[loopstart[1].v].co,
|
|
|
|
mvarray[loopstart[2].v].co,
|
|
|
|
mvarray[loopstart[3].v].co
|
|
|
|
);
|
2009-09-12 06:47:59 +00:00
|
|
|
}
|
2011-11-28 23:50:40 +00:00
|
|
|
else {
|
|
|
|
mesh_calc_ngon_center(mpoly, loopstart, mvarray, cent);
|
2009-09-12 06:47:59 +00:00
|
|
|
}
|
|
|
|
}
|
2010-09-04 05:31:25 +00:00
|
|
|
|
2011-11-29 02:58:38 +00:00
|
|
|
/* note, passing polynormal is only a speedup so we can skip calculating it */
|
2012-05-05 21:28:12 +00:00
|
|
|
float BKE_mesh_calc_poly_area(MPoly *mpoly, MLoop *loopstart,
|
2012-09-25 00:25:17 +00:00
|
|
|
MVert *mvarray, const float polynormal[3])
|
2011-11-29 02:58:38 +00:00
|
|
|
{
|
|
|
|
if (mpoly->totloop == 3) {
|
|
|
|
return area_tri_v3(mvarray[loopstart[0].v].co,
|
|
|
|
mvarray[loopstart[1].v].co,
|
|
|
|
mvarray[loopstart[2].v].co
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (mpoly->totloop == 4) {
|
|
|
|
return area_quad_v3(mvarray[loopstart[0].v].co,
|
|
|
|
mvarray[loopstart[1].v].co,
|
|
|
|
mvarray[loopstart[2].v].co,
|
|
|
|
mvarray[loopstart[3].v].co
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
int i;
|
2012-02-21 03:43:57 +00:00
|
|
|
MLoop *l_iter = loopstart;
|
2012-12-11 15:10:19 +00:00
|
|
|
float area, polynorm_local[3];
|
|
|
|
float (*vertexcos)[3] = BLI_array_alloca(vertexcos, mpoly->totloop);
|
2012-09-25 00:25:17 +00:00
|
|
|
const float *no = polynormal ? polynormal : polynorm_local;
|
2011-11-29 02:58:38 +00:00
|
|
|
|
|
|
|
/* pack vertex cos into an array for area_poly_v3 */
|
2012-02-21 03:43:57 +00:00
|
|
|
for (i = 0; i < mpoly->totloop; i++, l_iter++) {
|
|
|
|
copy_v3_v3(vertexcos[i], mvarray[l_iter->v].co);
|
2011-11-29 02:58:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* need normal for area_poly_v3 as well */
|
|
|
|
if (polynormal == NULL) {
|
2012-09-25 00:25:17 +00:00
|
|
|
BKE_mesh_calc_poly_normal(mpoly, loopstart, mvarray, polynorm_local);
|
2011-11-29 02:58:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* finally calculate the area */
|
|
|
|
area = area_poly_v3(mpoly->totloop, vertexcos, no);
|
|
|
|
|
|
|
|
return area;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-14 17:35:46 +00:00
|
|
|
/* note, results won't be correct if polygon is non-planar */
|
|
|
|
static float mesh_calc_poly_planar_area_centroid(MPoly *mpoly, MLoop *loopstart, MVert *mvarray, float cent[3])
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
float tri_area;
|
|
|
|
float total_area = 0.0f;
|
|
|
|
float v1[3], v2[3], v3[3], normal[3], tri_cent[3];
|
|
|
|
|
|
|
|
BKE_mesh_calc_poly_normal(mpoly, loopstart, mvarray, normal);
|
|
|
|
copy_v3_v3(v1, mvarray[loopstart[0].v].co);
|
|
|
|
copy_v3_v3(v2, mvarray[loopstart[1].v].co);
|
|
|
|
zero_v3(cent);
|
|
|
|
|
|
|
|
for (i = 2; i < mpoly->totloop; i++) {
|
|
|
|
copy_v3_v3(v3, mvarray[loopstart[i].v].co);
|
|
|
|
|
|
|
|
tri_area = area_tri_signed_v3(v1, v2, v3, normal);
|
|
|
|
total_area += tri_area;
|
|
|
|
|
|
|
|
cent_tri_v3(tri_cent, v1, v2, v3);
|
|
|
|
madd_v3_v3fl(cent, tri_cent, tri_area);
|
|
|
|
|
|
|
|
copy_v3_v3(v2, v3);
|
|
|
|
}
|
|
|
|
|
|
|
|
mul_v3_fl(cent, 1.0f / total_area);
|
|
|
|
|
|
|
|
return total_area;
|
|
|
|
}
|
|
|
|
|
2013-01-24 04:02:30 +00:00
|
|
|
/**
|
|
|
|
* This function takes the difference between 2 vertex-coord-arrays
|
|
|
|
* (\a vert_cos_src, \a vert_cos_dst),
|
|
|
|
* and applies the difference to \a vert_cos_new relative to \a vert_cos_org.
|
|
|
|
*
|
|
|
|
* \param vert_cos_src reference deform source.
|
|
|
|
* \param vert_cos_dst reference deform destination.
|
|
|
|
*
|
|
|
|
* \param vert_cos_org reference for the output location.
|
|
|
|
* \param vert_cos_new resulting coords.
|
|
|
|
*/
|
|
|
|
void BKE_mesh_calc_relative_deform(
|
|
|
|
const MPoly *mpoly, const int totpoly,
|
|
|
|
const MLoop *mloop, const int totvert,
|
|
|
|
|
|
|
|
const float (*vert_cos_src)[3],
|
|
|
|
const float (*vert_cos_dst)[3],
|
|
|
|
|
|
|
|
const float (*vert_cos_org)[3],
|
|
|
|
float (*vert_cos_new)[3])
|
|
|
|
{
|
|
|
|
const MPoly *mp;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
int *vert_accum = MEM_callocN(sizeof(*vert_accum) * totvert, __func__);
|
|
|
|
|
|
|
|
memset(vert_cos_new, '\0', sizeof(*vert_cos_new) * totvert);
|
|
|
|
|
|
|
|
for (i = 0, mp = mpoly; i < totpoly; i++, mp++) {
|
|
|
|
const MLoop *loopstart = mloop + mp->loopstart;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; j < mp->totloop; j++) {
|
|
|
|
int v_prev = (loopstart + ((mp->totloop + (j - 1)) % mp->totloop))->v;
|
|
|
|
int v_curr = (loopstart + j)->v;
|
|
|
|
int v_next = (loopstart + ((j + 1) % mp->totloop))->v;
|
|
|
|
|
|
|
|
float tvec[3];
|
|
|
|
|
|
|
|
barycentric_transform(
|
|
|
|
tvec, vert_cos_dst[v_curr],
|
|
|
|
vert_cos_org[v_prev], vert_cos_org[v_curr], vert_cos_org[v_next],
|
|
|
|
vert_cos_src[v_prev], vert_cos_src[v_curr], vert_cos_src[v_next]
|
|
|
|
);
|
|
|
|
|
|
|
|
add_v3_v3(vert_cos_new[v_curr], tvec);
|
|
|
|
vert_accum[v_curr] += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < totvert; i++) {
|
|
|
|
if (vert_accum[i]) {
|
|
|
|
mul_v3_fl(vert_cos_new[i], 1.0f / (float)vert_accum[i]);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
copy_v3_v3(vert_cos_new[i], vert_cos_org[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(vert_accum);
|
|
|
|
}
|
|
|
|
|
2012-02-28 23:08:40 +00:00
|
|
|
/* Find the index of the loop in 'poly' which references vertex,
|
2012-03-03 20:19:11 +00:00
|
|
|
* returns -1 if not found */
|
2012-02-28 23:08:40 +00:00
|
|
|
int poly_find_loop_from_vert(const MPoly *poly, const MLoop *loopstart,
|
2012-05-06 15:15:33 +00:00
|
|
|
unsigned vert)
|
2012-02-28 23:08:40 +00:00
|
|
|
{
|
|
|
|
int j;
|
|
|
|
for (j = 0; j < poly->totloop; j++, loopstart++) {
|
|
|
|
if (loopstart->v == vert)
|
|
|
|
return j;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fill 'adj_r' with the loop indices in 'poly' adjacent to the
|
2012-03-03 20:19:11 +00:00
|
|
|
* vertex. Returns the index of the loop matching vertex, or -1 if the
|
|
|
|
* vertex is not in 'poly' */
|
2012-02-28 23:08:40 +00:00
|
|
|
int poly_get_adj_loops_from_vert(unsigned adj_r[3], const MPoly *poly,
|
2012-05-06 15:15:33 +00:00
|
|
|
const MLoop *mloop, unsigned vert)
|
2012-02-28 23:08:40 +00:00
|
|
|
{
|
|
|
|
int corner = poly_find_loop_from_vert(poly,
|
2012-05-06 15:15:33 +00:00
|
|
|
&mloop[poly->loopstart],
|
|
|
|
vert);
|
2012-02-28 23:08:40 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (corner != -1) {
|
2012-02-28 23:08:40 +00:00
|
|
|
const MLoop *ml = &mloop[poly->loopstart + corner];
|
|
|
|
|
|
|
|
/* vertex was found */
|
|
|
|
adj_r[0] = ME_POLY_LOOP_PREV(mloop, poly, corner)->v;
|
|
|
|
adj_r[1] = ml->v;
|
|
|
|
adj_r[2] = ME_POLY_LOOP_NEXT(mloop, poly, corner)->v;
|
|
|
|
}
|
|
|
|
|
|
|
|
return corner;
|
|
|
|
}
|
|
|
|
|
2012-05-22 15:28:44 +00:00
|
|
|
/* Return the index of the edge vert that is not equal to 'v'. If
|
|
|
|
* neither edge vertex is equal to 'v', returns -1. */
|
|
|
|
int BKE_mesh_edge_other_vert(const MEdge *e, int v)
|
|
|
|
{
|
|
|
|
if (e->v1 == v)
|
|
|
|
return e->v2;
|
|
|
|
else if (e->v2 == v)
|
|
|
|
return e->v1;
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-03-14 06:31:14 +00:00
|
|
|
/* update the hide flag for edges and faces from the corresponding
|
2012-04-22 11:54:53 +00:00
|
|
|
* flag in verts */
|
2012-05-05 21:28:12 +00:00
|
|
|
void BKE_mesh_flush_hidden_from_verts(const MVert *mvert,
|
2012-05-06 15:15:33 +00:00
|
|
|
const MLoop *mloop,
|
|
|
|
MEdge *medge, int totedge,
|
|
|
|
MPoly *mpoly, int totpoly)
|
2012-03-14 06:31:14 +00:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
for (i = 0; i < totedge; i++) {
|
2012-03-14 06:31:14 +00:00
|
|
|
MEdge *e = &medge[i];
|
2012-03-24 06:18:31 +00:00
|
|
|
if (mvert[e->v1].flag & ME_HIDE ||
|
2012-04-28 06:31:57 +00:00
|
|
|
mvert[e->v2].flag & ME_HIDE)
|
|
|
|
{
|
2012-03-14 06:31:14 +00:00
|
|
|
e->flag |= ME_HIDE;
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-03-14 06:31:14 +00:00
|
|
|
e->flag &= ~ME_HIDE;
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2012-03-14 06:31:14 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
for (i = 0; i < totpoly; i++) {
|
2012-03-14 06:31:14 +00:00
|
|
|
MPoly *p = &mpoly[i];
|
|
|
|
p->flag &= ~ME_HIDE;
|
2012-03-24 06:18:31 +00:00
|
|
|
for (j = 0; j < p->totloop; j++) {
|
|
|
|
if (mvert[mloop[p->loopstart + j].v].flag & ME_HIDE)
|
2012-03-14 06:31:14 +00:00
|
|
|
p->flag |= ME_HIDE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-22 08:13:44 +00:00
|
|
|
/**
|
|
|
|
* simple poly -> vert/edge selection.
|
|
|
|
*/
|
2012-12-22 14:20:27 +00:00
|
|
|
void BKE_mesh_flush_select_from_polys_ex(MVert *mvert, const int totvert,
|
|
|
|
MLoop *mloop,
|
|
|
|
MEdge *medge, const int totedge,
|
|
|
|
const MPoly *mpoly, const int totpoly)
|
2012-12-22 08:13:44 +00:00
|
|
|
{
|
|
|
|
MVert *mv;
|
|
|
|
MEdge *med;
|
|
|
|
const MPoly *mp;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
i = totvert;
|
|
|
|
for (mv = mvert; i--; mv++) {
|
|
|
|
mv->flag &= ~SELECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
i = totedge;
|
|
|
|
for (med = medge; i--; med++) {
|
|
|
|
med->flag &= ~SELECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
i = totpoly;
|
|
|
|
for (mp = mpoly; i--; mp++) {
|
|
|
|
/* assume if its selected its not hidden and none of its verts/edges are hidden
|
|
|
|
* (a common assumption)*/
|
|
|
|
if (mp->flag & ME_FACE_SEL) {
|
|
|
|
MLoop *ml;
|
|
|
|
int j;
|
|
|
|
j = mp->totloop;
|
|
|
|
for (ml = &mloop[mp->loopstart]; j--; ml++) {
|
|
|
|
mvert[ml->v].flag |= SELECT;
|
|
|
|
medge[ml->e].flag |= SELECT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-12-22 14:20:27 +00:00
|
|
|
void BKE_mesh_flush_select_from_polys(Mesh *me)
|
|
|
|
{
|
|
|
|
BKE_mesh_flush_select_from_polys_ex(me->mvert, me->totvert,
|
|
|
|
me->mloop,
|
|
|
|
me->medge, me->totedge,
|
|
|
|
me->mpoly, me->totpoly);
|
|
|
|
}
|
2012-12-22 08:13:44 +00:00
|
|
|
|
2012-12-22 14:20:27 +00:00
|
|
|
void BKE_mesh_flush_select_from_verts_ex(const MVert *mvert, const int UNUSED(totvert),
|
|
|
|
MLoop *mloop,
|
|
|
|
MEdge *medge, const int totedge,
|
|
|
|
MPoly *mpoly, const int totpoly)
|
2012-12-22 13:42:06 +00:00
|
|
|
{
|
|
|
|
MEdge *med;
|
|
|
|
MPoly *mp;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* edges */
|
|
|
|
i = totedge;
|
|
|
|
for (med = medge; i--; med++) {
|
|
|
|
if ((med->flag & ME_HIDE) == 0) {
|
|
|
|
if ((mvert[med->v1].flag & SELECT) && (mvert[med->v2].flag & SELECT)) {
|
|
|
|
med->flag |= SELECT;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
med->flag &= ~SELECT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* polys */
|
|
|
|
i = totpoly;
|
|
|
|
for (mp = mpoly; i--; mp++) {
|
|
|
|
if ((mp->flag & ME_HIDE) == 0) {
|
|
|
|
int ok = TRUE;
|
|
|
|
MLoop *ml;
|
|
|
|
int j;
|
|
|
|
j = mp->totloop;
|
|
|
|
for (ml = &mloop[mp->loopstart]; j--; ml++) {
|
|
|
|
if ((mvert[ml->v].flag & SELECT) == 0) {
|
|
|
|
ok = FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ok) {
|
|
|
|
mp->flag |= ME_FACE_SEL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mp->flag &= ~ME_FACE_SEL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-12-22 14:20:27 +00:00
|
|
|
void BKE_mesh_flush_select_from_verts(Mesh *me)
|
|
|
|
{
|
|
|
|
BKE_mesh_flush_select_from_verts_ex(me->mvert, me->totvert,
|
|
|
|
me->mloop,
|
|
|
|
me->medge, me->totedge,
|
|
|
|
me->mpoly, me->totpoly);
|
|
|
|
}
|
|
|
|
|
2012-12-22 13:42:06 +00:00
|
|
|
|
2010-08-01 11:00:36 +00:00
|
|
|
/* basic vertex data functions */
|
2012-05-05 21:28:12 +00:00
|
|
|
int BKE_mesh_minmax(Mesh *me, float r_min[3], float r_max[3])
|
2010-08-01 11:00:36 +00:00
|
|
|
{
|
2012-05-05 21:28:12 +00:00
|
|
|
int i = me->totvert;
|
2010-08-01 11:00:36 +00:00
|
|
|
MVert *mvert;
|
2012-05-05 21:28:12 +00:00
|
|
|
for (mvert = me->mvert; i--; mvert++) {
|
2012-05-13 11:05:52 +00:00
|
|
|
minmax_v3v3_v3(r_min, r_max, mvert->co);
|
2010-08-01 11:00:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (me->totvert != 0);
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
int BKE_mesh_center_median(Mesh *me, float cent[3])
|
2010-08-01 11:00:36 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
int i = me->totvert;
|
2010-08-01 11:00:36 +00:00
|
|
|
MVert *mvert;
|
|
|
|
zero_v3(cent);
|
2012-05-06 15:15:33 +00:00
|
|
|
for (mvert = me->mvert; i--; mvert++) {
|
2010-08-01 11:00:36 +00:00
|
|
|
add_v3_v3(cent, mvert->co);
|
|
|
|
}
|
2011-02-09 15:32:39 +00:00
|
|
|
/* otherwise we get NAN for 0 verts */
|
2012-02-23 02:17:50 +00:00
|
|
|
if (me->totvert) {
|
2012-05-06 15:15:33 +00:00
|
|
|
mul_v3_fl(cent, 1.0f / (float)me->totvert);
|
2011-02-09 15:32:39 +00:00
|
|
|
}
|
2010-08-01 11:00:36 +00:00
|
|
|
|
|
|
|
return (me->totvert != 0);
|
|
|
|
}
|
|
|
|
|
2012-05-05 21:28:12 +00:00
|
|
|
int BKE_mesh_center_bounds(Mesh *me, float cent[3])
|
2010-08-01 11:00:36 +00:00
|
|
|
{
|
|
|
|
float min[3], max[3];
|
2010-08-03 00:56:43 +00:00
|
|
|
INIT_MINMAX(min, max);
|
2012-05-05 21:28:12 +00:00
|
|
|
if (BKE_mesh_minmax(me, min, max)) {
|
2010-08-01 11:00:36 +00:00
|
|
|
mid_v3_v3v3(cent, min, max);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-10-20 16:48:54 +00:00
|
|
|
int BKE_mesh_center_centroid(Mesh *me, float cent[3])
|
|
|
|
{
|
|
|
|
int i = me->totpoly;
|
|
|
|
MPoly *mpoly;
|
|
|
|
float poly_area;
|
|
|
|
float total_area = 0.0f;
|
|
|
|
float poly_cent[3];
|
|
|
|
|
|
|
|
zero_v3(cent);
|
|
|
|
|
|
|
|
/* calculate a weighted average of polygon centroids */
|
|
|
|
for (mpoly = me->mpoly; i--; mpoly++) {
|
2013-02-14 17:35:46 +00:00
|
|
|
poly_area = mesh_calc_poly_planar_area_centroid(mpoly, me->mloop + mpoly->loopstart, me->mvert, poly_cent);
|
|
|
|
|
2012-10-20 16:48:54 +00:00
|
|
|
madd_v3_v3fl(cent, poly_cent, poly_area);
|
|
|
|
total_area += poly_area;
|
|
|
|
}
|
|
|
|
/* otherwise we get NAN for 0 polys */
|
|
|
|
if (me->totpoly) {
|
|
|
|
mul_v3_fl(cent, 1.0f / total_area);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (me->totpoly != 0);
|
|
|
|
}
|
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
void BKE_mesh_translate(Mesh *me, const float offset[3], const bool do_keys)
|
2010-08-01 11:00:36 +00:00
|
|
|
{
|
2012-05-06 15:15:33 +00:00
|
|
|
int i = me->totvert;
|
2010-08-01 11:00:36 +00:00
|
|
|
MVert *mvert;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (mvert = me->mvert; i--; mvert++) {
|
2010-08-01 11:00:36 +00:00
|
|
|
add_v3_v3(mvert->co, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (do_keys && me->key) {
|
|
|
|
KeyBlock *kb;
|
2012-05-06 15:15:33 +00:00
|
|
|
for (kb = me->key->block.first; kb; kb = kb->next) {
|
|
|
|
float *fp = kb->data;
|
|
|
|
for (i = kb->totelem; i--; fp += 3) {
|
2010-08-01 11:00:36 +00:00
|
|
|
add_v3_v3(fp, offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-10-09 21:11:51 +00:00
|
|
|
|
|
|
|
void BKE_mesh_ensure_navmesh(Mesh *me)
|
|
|
|
{
|
2012-02-13 04:52:41 +00:00
|
|
|
if (!CustomData_has_layer(&me->pdata, CD_RECAST)) {
|
2011-10-09 21:11:51 +00:00
|
|
|
int i;
|
2012-02-13 04:52:41 +00:00
|
|
|
int numFaces = me->totpoly;
|
2012-03-24 02:51:46 +00:00
|
|
|
int *recastData;
|
2012-07-03 08:16:14 +00:00
|
|
|
recastData = (int *)MEM_mallocN(numFaces * sizeof(int), __func__);
|
2012-05-06 15:15:33 +00:00
|
|
|
for (i = 0; i < numFaces; i++) {
|
|
|
|
recastData[i] = i + 1;
|
2011-10-09 21:11:51 +00:00
|
|
|
}
|
2012-07-03 08:16:14 +00:00
|
|
|
CustomData_add_layer_named(&me->pdata, CD_RECAST, CD_ASSIGN, recastData, numFaces, "recastData");
|
2011-10-09 21:11:51 +00:00
|
|
|
}
|
|
|
|
}
|
2012-02-05 07:12:46 +00:00
|
|
|
|
2012-02-12 19:11:09 +00:00
|
|
|
void BKE_mesh_tessface_calc(Mesh *mesh)
|
2012-02-05 07:12:46 +00:00
|
|
|
{
|
2012-05-05 21:28:12 +00:00
|
|
|
mesh->totface = BKE_mesh_recalc_tessellation(&mesh->fdata, &mesh->ldata, &mesh->pdata,
|
2012-05-06 15:15:33 +00:00
|
|
|
mesh->mvert,
|
|
|
|
mesh->totface, mesh->totloop, mesh->totpoly,
|
|
|
|
/* calc normals right after, don't copy from polys here */
|
2013-03-17 19:55:10 +00:00
|
|
|
false);
|
2012-02-05 07:12:46 +00:00
|
|
|
|
2013-03-17 19:55:10 +00:00
|
|
|
BKE_mesh_update_customdata_pointers(mesh, true);
|
2012-02-05 07:12:46 +00:00
|
|
|
}
|
|
|
|
|
2012-02-12 19:11:09 +00:00
|
|
|
void BKE_mesh_tessface_ensure(Mesh *mesh)
|
2012-02-05 07:12:46 +00:00
|
|
|
{
|
|
|
|
if (mesh->totpoly && mesh->totface == 0) {
|
2012-02-12 19:11:09 +00:00
|
|
|
BKE_mesh_tessface_calc(mesh);
|
2012-02-05 07:12:46 +00:00
|
|
|
}
|
|
|
|
}
|
2012-02-08 11:52:44 +00:00
|
|
|
|
2012-02-12 19:11:09 +00:00
|
|
|
void BKE_mesh_tessface_clear(Mesh *mesh)
|
2012-02-08 11:52:44 +00:00
|
|
|
{
|
2012-03-06 16:22:41 +00:00
|
|
|
mesh_tessface_clear_intern(mesh, TRUE);
|
2012-02-08 11:52:44 +00:00
|
|
|
}
|
2012-05-08 14:58:38 +00:00
|
|
|
|
|
|
|
#if 0 /* slow version of the function below */
|
|
|
|
void BKE_mesh_poly_calc_angles(MVert *mvert, MLoop *mloop,
|
|
|
|
MPoly *mp, float angles[])
|
|
|
|
{
|
|
|
|
MLoop *ml;
|
|
|
|
|
|
|
|
int j;
|
|
|
|
for (j = 0, ml = mloop + mp->loopstart; j < mp->totloop; j++, ml++) {
|
|
|
|
MLoop *ml_prev = ME_POLY_LOOP_PREV(mloop, mp, j);
|
|
|
|
MLoop *ml_next = ME_POLY_LOOP_NEXT(mloop, mp, j);
|
|
|
|
|
|
|
|
float e1[3], e2[3];
|
|
|
|
|
|
|
|
sub_v3_v3v3(e1, mvert[ml_next->v].co, mvert[ml->v].co);
|
|
|
|
sub_v3_v3v3(e2, mvert[ml_prev->v].co, mvert[ml->v].co);
|
|
|
|
|
|
|
|
angles[j] = (float)M_PI - angle_v3v3(e1, e2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* equivalent the function above but avoid multiple subtractions + normalize */
|
|
|
|
|
|
|
|
void BKE_mesh_poly_calc_angles(MVert *mvert, MLoop *mloop,
|
|
|
|
MPoly *mp, float angles[])
|
|
|
|
{
|
|
|
|
MLoop *ml = mloop + mp->loopstart;
|
|
|
|
float nor_prev[3];
|
|
|
|
float nor_next[3];
|
|
|
|
|
|
|
|
int i_this = mp->totloop - 1;
|
|
|
|
int i_next = 0;
|
|
|
|
|
|
|
|
sub_v3_v3v3(nor_prev, mvert[ml[i_this - 1].v].co, mvert[ml[i_this].v].co);
|
|
|
|
normalize_v3(nor_prev);
|
|
|
|
|
|
|
|
while (i_next < mp->totloop) {
|
|
|
|
sub_v3_v3v3(nor_next, mvert[ml[i_this].v].co, mvert[ml[i_next].v].co);
|
|
|
|
normalize_v3(nor_next);
|
|
|
|
angles[i_this] = angle_normalized_v3v3(nor_prev, nor_next);
|
|
|
|
|
|
|
|
/* step */
|
|
|
|
copy_v3_v3(nor_prev, nor_next);
|
|
|
|
i_this = i_next;
|
|
|
|
i_next++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2013-01-10 04:43:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
void BKE_mesh_do_versions_cd_flag_init(Mesh *mesh)
|
|
|
|
{
|
|
|
|
if (UNLIKELY(mesh->cd_flag)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
MVert *mv;
|
|
|
|
MEdge *med;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (mv = mesh->mvert, i = 0; i < mesh->totvert; mv++, i++) {
|
|
|
|
if (mv->bweight != 0) {
|
|
|
|
mesh->cd_flag |= ME_CDFLAG_VERT_BWEIGHT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (med = mesh->medge, i = 0; i < mesh->totedge; med++, i++) {
|
|
|
|
if (med->bweight != 0) {
|
|
|
|
mesh->cd_flag |= ME_CDFLAG_EDGE_BWEIGHT;
|
|
|
|
if (mesh->cd_flag & ME_CDFLAG_EDGE_CREASE) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (med->crease != 0) {
|
|
|
|
mesh->cd_flag |= ME_CDFLAG_EDGE_CREASE;
|
|
|
|
if (mesh->cd_flag & ME_CDFLAG_EDGE_BWEIGHT) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|