main features are: * Modifiers can now be in any order in the modifier stack * DerivedMesh now has a standard framework for custom element data to be passed through the stack with mesh data (being copied and interpolated as appropriate), so modifiers can access whatever data they need * The modifier stack code has been refactored and a number of bugs have been removed * The EdgeSplit modifier has been added: http://mediawiki.blender.org/index.php/BlenderDev/EdgeSplitModifier * The DerivedMesh modifier has been added: http://mediawiki.blender.org/index.php/BlenderDev/DisplaceModifier * The UVProject modifier has been added: http://mediawiki.blender.org/index.php/BlenderDev/UVProjectModifier For more info, see: http://mediawiki.blender.org/index.php/User:Artificer/ModifierStackUpgrade (currently undergoing reorganisation)
3276 lines
94 KiB
C
3276 lines
94 KiB
C
/**
|
|
* $Id$
|
|
*
|
|
* ***** BEGIN GPL/BL DUAL LICENSE BLOCK *****
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version. The Blender
|
|
* Foundation also sells licenses for use in proprietary software under
|
|
* the Blender License. See http://www.blender.org/BL/ for information
|
|
* about this.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*
|
|
* The Original Code is Copyright (C) 2005 Blender Foundation.
|
|
* All rights reserved.
|
|
*
|
|
* The Original Code is: all of this file.
|
|
*
|
|
* Contributor(s): none yet.
|
|
*
|
|
* ***** END GPL/BL DUAL LICENSE BLOCK *****
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <math.h>
|
|
#include <float.h>
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "DNA_mesh_types.h"
|
|
#include "DNA_meshdata_types.h"
|
|
#include "DNA_modifier_types.h"
|
|
#include "DNA_object_types.h"
|
|
|
|
#include "BKE_bad_level_calls.h"
|
|
#include "BKE_cdderivedmesh.h"
|
|
#include "BKE_utildefines.h"
|
|
#include "BKE_global.h"
|
|
#include "BKE_mesh.h"
|
|
#include "BKE_subsurf.h"
|
|
#include "BKE_displist.h"
|
|
#include "BKE_DerivedMesh.h"
|
|
|
|
#include "BLI_blenlib.h"
|
|
#include "BLI_editVert.h"
|
|
#include "BLI_arithb.h"
|
|
#include "BLI_linklist.h"
|
|
#include "BLI_memarena.h"
|
|
#include "BLI_edgehash.h"
|
|
|
|
#include "BIF_gl.h"
|
|
|
|
#include "CCGSubSurf.h"
|
|
|
|
typedef struct _VertData {
|
|
float co[3];
|
|
float no[3];
|
|
} VertData;
|
|
|
|
struct CCGDerivedMesh {
|
|
DerivedMesh dm;
|
|
|
|
CCGSubSurf *ss;
|
|
int fromEditmesh, drawInteriorEdges, useSubsurfUv;
|
|
|
|
Mesh *me;
|
|
DispListMesh *dlm;
|
|
|
|
struct {int startVert; CCGVert *vert;} *vertMap;
|
|
struct {int startVert; int startEdge; CCGEdge *edge;} *edgeMap;
|
|
struct {int startVert; int startEdge;
|
|
int startFace; CCGFace *face;} *faceMap;
|
|
};
|
|
|
|
typedef struct CCGDerivedMesh CCGDerivedMesh;
|
|
|
|
static int ccgDM_getVertMapIndex(CCGDerivedMesh *ccgdm, CCGSubSurf *ss, CCGVert *v);
|
|
static int ccgDM_getEdgeMapIndex(CCGDerivedMesh *ccgdm, CCGSubSurf *ss, CCGEdge *e);
|
|
static int ccgDM_getFaceMapIndex(CCGDerivedMesh *ccgdm, CCGSubSurf *ss, CCGFace *f);
|
|
|
|
///
|
|
|
|
static void *arena_alloc(CCGAllocatorHDL a, int numBytes) {
|
|
return BLI_memarena_alloc(a, numBytes);
|
|
}
|
|
static void *arena_realloc(CCGAllocatorHDL a, void *ptr, int newSize, int oldSize) {
|
|
void *p2 = BLI_memarena_alloc(a, newSize);
|
|
if (ptr) {
|
|
memcpy(p2, ptr, oldSize);
|
|
}
|
|
return p2;
|
|
}
|
|
static void arena_free(CCGAllocatorHDL a, void *ptr) {
|
|
}
|
|
static void arena_release(CCGAllocatorHDL a) {
|
|
BLI_memarena_free(a);
|
|
}
|
|
|
|
static CCGSubSurf *_getSubSurf(CCGSubSurf *prevSS, int subdivLevels, int useAging, int useArena, int useFlatSubdiv) {
|
|
CCGMeshIFC ifc;
|
|
CCGSubSurf *ccgSS;
|
|
|
|
/* subdivLevels==0 is not allowed */
|
|
subdivLevels = MAX2(subdivLevels, 1);
|
|
|
|
if (prevSS) {
|
|
int oldUseAging;
|
|
|
|
useAging = !!useAging;
|
|
ccgSubSurf_getUseAgeCounts(prevSS, &oldUseAging, NULL, NULL, NULL);
|
|
|
|
if (oldUseAging!=useAging) {
|
|
ccgSubSurf_free(prevSS);
|
|
} else {
|
|
ccgSubSurf_setSubdivisionLevels(prevSS, subdivLevels);
|
|
|
|
return prevSS;
|
|
}
|
|
}
|
|
|
|
if (useAging) {
|
|
ifc.vertUserSize = ifc.edgeUserSize = ifc.faceUserSize = 12;
|
|
} else {
|
|
ifc.vertUserSize = ifc.edgeUserSize = ifc.faceUserSize = 8;
|
|
}
|
|
ifc.vertDataSize = sizeof(VertData);
|
|
|
|
if (useArena) {
|
|
CCGAllocatorIFC allocatorIFC;
|
|
CCGAllocatorHDL allocator = BLI_memarena_new((1<<16));
|
|
|
|
allocatorIFC.alloc = arena_alloc;
|
|
allocatorIFC.realloc = arena_realloc;
|
|
allocatorIFC.free = arena_free;
|
|
allocatorIFC.release = arena_release;
|
|
|
|
ccgSS = ccgSubSurf_new(&ifc, subdivLevels, &allocatorIFC, allocator);
|
|
} else {
|
|
ccgSS = ccgSubSurf_new(&ifc, subdivLevels, NULL, NULL);
|
|
}
|
|
|
|
if (useAging) {
|
|
ccgSubSurf_setUseAgeCounts(ccgSS, 1, 8, 8, 8);
|
|
}
|
|
|
|
ccgSubSurf_setCalcVertexNormals(ccgSS, 1, BLI_STRUCT_OFFSET(VertData, no));
|
|
|
|
return ccgSS;
|
|
}
|
|
|
|
static int getEdgeIndex(CCGSubSurf *ss, CCGEdge *e, int x, int edgeSize) {
|
|
CCGVert *v0 = ccgSubSurf_getEdgeVert0(ss, e);
|
|
CCGVert *v1 = ccgSubSurf_getEdgeVert1(ss, e);
|
|
int v0idx = *((int*) ccgSubSurf_getVertUserData(ss, v0));
|
|
int v1idx = *((int*) ccgSubSurf_getVertUserData(ss, v1));
|
|
int edgeBase = *((int*) ccgSubSurf_getEdgeUserData(ss, e));
|
|
|
|
if (x==0) {
|
|
return v0idx;
|
|
} else if (x==edgeSize-1) {
|
|
return v1idx;
|
|
} else {
|
|
return edgeBase + x-1;
|
|
}
|
|
}
|
|
static int getFaceIndex(CCGSubSurf *ss, CCGFace *f, int S, int x, int y, int edgeSize, int gridSize) {
|
|
int faceBase = *((int*) ccgSubSurf_getFaceUserData(ss, f));
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
if (x==gridSize-1 && y==gridSize-1) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
return *((int*) ccgSubSurf_getVertUserData(ss, v));
|
|
} else if (x==gridSize-1) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
CCGEdge *e = ccgSubSurf_getFaceEdge(ss, f, S);
|
|
int edgeBase = *((int*) ccgSubSurf_getEdgeUserData(ss, e));
|
|
if (v==ccgSubSurf_getEdgeVert0(ss, e)) {
|
|
return edgeBase + (gridSize-1-y)-1;
|
|
} else {
|
|
return edgeBase + (edgeSize-2-1)-((gridSize-1-y)-1);
|
|
}
|
|
} else if (y==gridSize-1) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
CCGEdge *e = ccgSubSurf_getFaceEdge(ss, f, (S+numVerts-1)%numVerts);
|
|
int edgeBase = *((int*) ccgSubSurf_getEdgeUserData(ss, e));
|
|
if (v==ccgSubSurf_getEdgeVert0(ss, e)) {
|
|
return edgeBase + (gridSize-1-x)-1;
|
|
} else {
|
|
return edgeBase + (edgeSize-2-1)-((gridSize-1-x)-1);
|
|
}
|
|
} else if (x==0 && y==0) {
|
|
return faceBase;
|
|
} else if (x==0) {
|
|
S = (S+numVerts-1)%numVerts;
|
|
return faceBase + 1 + (gridSize-2)*S + (y-1);
|
|
} else if (y==0) {
|
|
return faceBase + 1 + (gridSize-2)*S + (x-1);
|
|
} else {
|
|
return faceBase + 1 + (gridSize-2)*numVerts + S*(gridSize-2)*(gridSize-2) + (y-1)*(gridSize-2) + (x-1);
|
|
}
|
|
}
|
|
|
|
static float *getFaceUV(CCGSubSurf *ss, CCGFace *f, int S, int x, int y, int edgeSize, int gridSize)
|
|
{
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
if (x==gridSize-1 && y==gridSize-1) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
return ccgSubSurf_getVertData(ss, v);
|
|
}
|
|
else if (x==gridSize-1) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
CCGEdge *e = ccgSubSurf_getFaceEdge(ss, f, S);
|
|
|
|
if (v==ccgSubSurf_getEdgeVert0(ss, e))
|
|
return ccgSubSurf_getEdgeData(ss, e, gridSize-1-y);
|
|
else
|
|
return ccgSubSurf_getEdgeData(ss, e, (edgeSize-2-1)-(gridSize-1-y-2));
|
|
}
|
|
else if (y==gridSize-1) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
CCGEdge *e = ccgSubSurf_getFaceEdge(ss, f, (S+numVerts-1)%numVerts);
|
|
|
|
if (v==ccgSubSurf_getEdgeVert0(ss, e))
|
|
return ccgSubSurf_getEdgeData(ss, e, gridSize-1-x);
|
|
else
|
|
return ccgSubSurf_getEdgeData(ss, e, (edgeSize-2-1)-(gridSize-1-x-2));
|
|
}
|
|
else if (x==0 && y==0)
|
|
return ccgSubSurf_getFaceCenterData(ss, f);
|
|
else if (x==0)
|
|
return ccgSubSurf_getFaceGridEdgeData(ss, f, (S+numVerts-1)%numVerts, y);
|
|
else if (y==0)
|
|
return ccgSubSurf_getFaceGridEdgeData(ss, f, S, x);
|
|
else
|
|
return ccgSubSurf_getFaceGridData(ss, f, S, x, y);
|
|
}
|
|
|
|
static void get_face_uv_map_vert(UvVertMap *vmap, struct MFace *mf, int fi, CCGVertHDL *fverts) {
|
|
unsigned int *fv = &mf->v1;
|
|
UvMapVert *v, *nv;
|
|
int j, nverts= mf->v4? 4: 3;
|
|
|
|
for (j=0; j<nverts; j++, fv++) {
|
|
for (nv=v=get_uv_map_vert(vmap, *fv); v; v=v->next) {
|
|
if (v->separate)
|
|
nv= v;
|
|
if (v->f == fi)
|
|
break;
|
|
}
|
|
|
|
fverts[j]= (CCGVertHDL)(nv->f*4 + nv->tfindex);
|
|
}
|
|
}
|
|
|
|
static int ss_sync_from_uv(CCGSubSurf *ss, CCGSubSurf *origss, Mesh *me, DispListMesh *dlm) {
|
|
MFace *mface = dlm?dlm->mface:me->mface;
|
|
TFace *tface = dlm?dlm->tface:me->tface;
|
|
MVert *mvert = dlm?dlm->mvert:me->mvert;
|
|
int totvert = dlm?dlm->totvert:me->totvert;
|
|
int totface = dlm?dlm->totface:me->totface;
|
|
int i, j, seam;
|
|
UvMapVert *v;
|
|
UvVertMap *vmap;
|
|
float limit[2];
|
|
CCGVertHDL fverts[4];
|
|
EdgeHash *ehash;
|
|
float creaseFactor = (float)ccgSubSurf_getSubdivisionLevels(ss);
|
|
|
|
limit[0]= limit[1]= 0.0001f;
|
|
vmap= make_uv_vert_map(mface, tface, totface, totvert, 0, limit);
|
|
if (!vmap)
|
|
return 0;
|
|
|
|
ccgSubSurf_initFullSync(ss);
|
|
|
|
/* create vertices */
|
|
for (i=0; i<totvert; i++) {
|
|
if (!get_uv_map_vert(vmap, i))
|
|
continue;
|
|
|
|
for (v=get_uv_map_vert(vmap, i)->next; v; v=v->next)
|
|
if (v->separate)
|
|
break;
|
|
|
|
seam = (v != NULL) || ((mvert+i)->flag & ME_VERT_MERGED);
|
|
|
|
for (v=get_uv_map_vert(vmap, i); v; v=v->next) {
|
|
if (v->separate) {
|
|
CCGVert *ssv;
|
|
CCGVertHDL vhdl = (CCGVertHDL)(v->f*4 + v->tfindex);
|
|
float uv[3];
|
|
|
|
uv[0]= (tface+v->f)->uv[v->tfindex][0];
|
|
uv[1]= (tface+v->f)->uv[v->tfindex][1];
|
|
uv[2]= 0.0f;
|
|
|
|
ccgSubSurf_syncVert(ss, vhdl, uv, seam, &ssv);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* create edges */
|
|
ehash = BLI_edgehash_new();
|
|
|
|
for (i=0; i<totface; i++) {
|
|
MFace *mf = &((MFace*) mface)[i];
|
|
int nverts= mf->v4? 4: 3;
|
|
CCGFace *origf= ccgSubSurf_getFace(origss, (CCGFaceHDL)i);
|
|
unsigned int *fv = &mf->v1;
|
|
|
|
get_face_uv_map_vert(vmap, mf, i, fverts);
|
|
|
|
for (j=0; j<nverts; j++) {
|
|
int v0 = (int)fverts[j];
|
|
int v1 = (int)fverts[(j+1)%nverts];
|
|
MVert *mv0 = mvert + *(fv+j);
|
|
MVert *mv1 = mvert + *(fv+((j+1)%nverts));
|
|
|
|
if (!BLI_edgehash_haskey(ehash, v0, v1)) {
|
|
CCGEdge *e, *orige= ccgSubSurf_getFaceEdge(origss, origf, j);
|
|
CCGEdgeHDL ehdl= (CCGEdgeHDL)(i*4 + j);
|
|
float crease;
|
|
|
|
if ((mv0->flag&mv1->flag) & ME_VERT_MERGED)
|
|
crease = creaseFactor;
|
|
else
|
|
crease = ccgSubSurf_getEdgeCrease(origss, orige);
|
|
|
|
ccgSubSurf_syncEdge(ss, ehdl, fverts[j], fverts[(j+1)%nverts], crease, &e);
|
|
BLI_edgehash_insert(ehash, v0, v1, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
BLI_edgehash_free(ehash, NULL);
|
|
|
|
/* create faces */
|
|
for (i=0; i<totface; i++) {
|
|
MFace *mf = &((MFace*) mface)[i];
|
|
int nverts= mf->v4? 4: 3;
|
|
CCGFace *f;
|
|
|
|
get_face_uv_map_vert(vmap, mf, i, fverts);
|
|
ccgSubSurf_syncFace(ss, (CCGFaceHDL)i, nverts, fverts, &f);
|
|
}
|
|
|
|
free_uv_vert_map(vmap);
|
|
ccgSubSurf_processSync(ss);
|
|
|
|
return 1;
|
|
}
|
|
|
|
#if 0
|
|
static unsigned int ss_getEdgeFlags(CCGSubSurf *ss, CCGEdge *e, int ssFromEditmesh, DispListMesh *dlm, MEdge *medge, TFace *tface)
|
|
{
|
|
unsigned int flags = 0;
|
|
int N = ccgSubSurf_getEdgeNumFaces(ss, e);
|
|
|
|
if (!N) flags |= ME_LOOSEEDGE;
|
|
|
|
if (ssFromEditmesh) {
|
|
EditEdge *eed = ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
flags |= ME_EDGEDRAW|ME_EDGERENDER;
|
|
if (eed->seam) {
|
|
flags |= ME_SEAM;
|
|
}
|
|
} else {
|
|
if (edgeIdx!=-1) {
|
|
MEdge *origMed = &medge[edgeIdx];
|
|
|
|
if (dlm) {
|
|
flags |= origMed->flag&~ME_EDGE_STEPINDEX;
|
|
} else {
|
|
flags |= (origMed->flag&ME_SEAM)|ME_EDGEDRAW|ME_EDGERENDER;
|
|
}
|
|
}
|
|
}
|
|
|
|
return flags;
|
|
}
|
|
#endif
|
|
|
|
static DispListMesh *ss_to_displistmesh(CCGSubSurf *ss, CCGDerivedMesh *ccgdm, int ssFromEditmesh, int drawInteriorEdges, int useSubsurfUv, Mesh *inMe, DispListMesh *inDLM) {
|
|
DispListMesh *dlm = MEM_callocN(sizeof(*dlm), "dlm");
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int edgeBase, faceBase;
|
|
int i, j, k, S, x, y, index;
|
|
int vertBase = 0;
|
|
TFace *tface = NULL;
|
|
MEdge *medge = NULL;
|
|
MFace *mface = NULL;
|
|
MCol *mcol = NULL;
|
|
CCGVertIterator *vi;
|
|
CCGEdgeIterator *ei;
|
|
CCGFaceIterator *fi;
|
|
CCGFace **faceMap2, **faceMap2Uv = NULL;
|
|
CCGEdge **edgeMap2;
|
|
CCGVert **vertMap2;
|
|
int totvert, totedge, totface;
|
|
CCGSubSurf *uvss= NULL;
|
|
|
|
totvert = ccgSubSurf_getNumVerts(ss);
|
|
vertMap2 = MEM_mallocN(totvert*sizeof(*vertMap2), "vertmap");
|
|
vi = ccgSubSurf_getVertIterator(ss);
|
|
for (; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
|
|
if (ssFromEditmesh) {
|
|
vertMap2[ccgDM_getVertMapIndex(ccgdm,ss,v)] = v;
|
|
} else {
|
|
vertMap2[(int) ccgSubSurf_getVertVertHandle(ss, v)] = v;
|
|
}
|
|
}
|
|
ccgVertIterator_free(vi);
|
|
|
|
totedge = ccgSubSurf_getNumEdges(ss);
|
|
edgeMap2 = MEM_mallocN(totedge*sizeof(*edgeMap2), "edgemap");
|
|
ei = ccgSubSurf_getEdgeIterator(ss);
|
|
for (i=0; !ccgEdgeIterator_isStopped(ei); i++,ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
|
|
if (ssFromEditmesh) {
|
|
edgeMap2[ccgDM_getEdgeMapIndex(ccgdm,ss,e)] = e;
|
|
} else {
|
|
edgeMap2[(int) ccgSubSurf_getEdgeEdgeHandle(ss, e)] = e;
|
|
}
|
|
}
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
faceMap2 = MEM_mallocN(totface*sizeof(*faceMap2), "facemap");
|
|
fi = ccgSubSurf_getFaceIterator(ss);
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
|
|
if (ssFromEditmesh) {
|
|
faceMap2[ccgDM_getFaceMapIndex(ccgdm,ss,f)] = f;
|
|
} else {
|
|
faceMap2[(int) ccgSubSurf_getFaceFaceHandle(ss, f)] = f;
|
|
}
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
|
|
if (!ssFromEditmesh) {
|
|
if (inDLM) {
|
|
tface = inDLM->tface;
|
|
medge = inDLM->medge;
|
|
mface = inDLM->mface;
|
|
mcol = inDLM->mcol;
|
|
} else if (inMe) {
|
|
tface = inMe->tface;
|
|
medge = inMe->medge;
|
|
mface = inMe->mface;
|
|
mcol = inMe->mcol;
|
|
}
|
|
}
|
|
|
|
dlm->totvert = ccgSubSurf_getNumFinalVerts(ss);
|
|
dlm->totedge = ccgSubSurf_getNumFinalEdges(ss);
|
|
dlm->totface = ccgSubSurf_getNumFinalFaces(ss);
|
|
|
|
dlm->mvert = MEM_callocN(dlm->totvert*sizeof(*dlm->mvert), "dlm->mvert");
|
|
dlm->medge = MEM_callocN(dlm->totedge*sizeof(*dlm->medge), "dlm->medge");
|
|
dlm->mface = MEM_callocN(dlm->totface*sizeof(*dlm->mface), "dlm->mface");
|
|
if (!ssFromEditmesh && tface) {
|
|
dlm->tface = MEM_callocN(dlm->totface*sizeof(*dlm->tface), "dlm->tface");
|
|
dlm->mcol = NULL;
|
|
} else if (!ssFromEditmesh && mcol) {
|
|
dlm->tface = NULL;
|
|
dlm->mcol = MEM_mallocN(dlm->totface*4*sizeof(*dlm->mcol), "dlm->mcol");
|
|
} else {
|
|
dlm->tface = NULL;
|
|
dlm->mcol = NULL;
|
|
}
|
|
|
|
|
|
if (useSubsurfUv && tface) {
|
|
/* not for editmesh currently */
|
|
uvss= _getSubSurf(NULL, ccgSubSurf_getSubdivisionLevels(ss), 0, 1, 0);
|
|
|
|
if (ss_sync_from_uv(uvss, ss, inMe, inDLM)) {
|
|
faceMap2Uv = MEM_mallocN(totface*sizeof(*faceMap2Uv), "facemapuv");
|
|
|
|
fi = ccgSubSurf_getFaceIterator(uvss);
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
faceMap2Uv[(int) ccgSubSurf_getFaceFaceHandle(uvss, f)] = f;
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
}
|
|
|
|
/* Load vertices... we do in this funny order because
|
|
* all "added" vertices" are required to appear first
|
|
* in the displist (before STEPINDEX flags start). Also
|
|
* note that the vertex with index 0 is always a face
|
|
* center vert, this is relied upon to ensure we don't
|
|
* need to do silly test_index_face calls.
|
|
*
|
|
* NOTE (artificer): The STEPINDEX flag has been removed, so this
|
|
* funny order is no longer strictly necessary, but it works.
|
|
*/
|
|
|
|
faceBase = i = 0;
|
|
for (index=0; index<totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
VecCopyf(dlm->mvert[i++].co, ccgSubSurf_getFaceCenterData(ss, f));
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
for (x=1; x<gridSize-1; x++) {
|
|
VecCopyf(dlm->mvert[i++].co, ccgSubSurf_getFaceGridEdgeData(ss, f, S, x));
|
|
}
|
|
}
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
for (y=1; y<gridSize-1; y++) {
|
|
for (x=1; x<gridSize-1; x++) {
|
|
VecCopyf(dlm->mvert[i++].co, ccgSubSurf_getFaceGridData(ss, f, S, x, y));
|
|
}
|
|
}
|
|
}
|
|
|
|
*((int*) ccgSubSurf_getFaceUserData(ss, f)) = faceBase;
|
|
faceBase += 1 + numVerts*((gridSize-2) + (gridSize-2)*(gridSize-2));
|
|
}
|
|
|
|
edgeBase = i;
|
|
for (index=0; index<totedge; index++) {
|
|
CCGEdge *e= edgeMap2[index];
|
|
int x;
|
|
|
|
for (x=1; x<edgeSize-1; x++) {
|
|
VecCopyf(dlm->mvert[i++].co, ccgSubSurf_getEdgeData(ss, e, x));
|
|
}
|
|
|
|
*((int*) ccgSubSurf_getEdgeUserData(ss, e)) = edgeBase;
|
|
edgeBase += edgeSize-2;
|
|
}
|
|
|
|
vertBase = i;
|
|
for (index=0; index<totvert; index++) {
|
|
CCGVert *v = vertMap2[index];
|
|
VecCopyf(dlm->mvert[i].co, ccgSubSurf_getVertData(ss, v));
|
|
*((int*) ccgSubSurf_getVertUserData(ss, v)) = i++;
|
|
}
|
|
|
|
// load edges
|
|
|
|
i = 0;
|
|
for (index=0; index<totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
for (k=0; k<numVerts; k++) {
|
|
for (x=0; x<gridSize-1; x++) {
|
|
MEdge *med = &dlm->medge[i];
|
|
if (drawInteriorEdges) med->flag = ME_EDGEDRAW|ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, k, x, 0, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, k, x+1, 0, edgeSize, gridSize);
|
|
i++;
|
|
}
|
|
|
|
for (x=1; x<gridSize-1; x++) {
|
|
for (y=0; y<gridSize-1; y++) {
|
|
MEdge *med;
|
|
|
|
med = &dlm->medge[i];
|
|
if (drawInteriorEdges) med->flag = ME_EDGEDRAW|ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, k, x, y, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, k, x, y+1, edgeSize, gridSize);
|
|
i++;
|
|
|
|
med = &dlm->medge[i];
|
|
if (drawInteriorEdges) med->flag = ME_EDGEDRAW|ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, k, y, x, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, k, y+1, x, edgeSize, gridSize);
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (index=0; index<totedge; index++) {
|
|
CCGEdge *e= edgeMap2[index];
|
|
unsigned int flags = 0;
|
|
|
|
if (!ccgSubSurf_getEdgeNumFaces(ss, e)) flags |= ME_LOOSEEDGE;
|
|
|
|
if (ssFromEditmesh) {
|
|
EditEdge *eed = ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
flags |= ME_EDGEDRAW|ME_EDGERENDER;
|
|
if (eed->seam) {
|
|
flags |= ME_SEAM;
|
|
}
|
|
} else if(medge){
|
|
int edgeIdx = (int) ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
if (edgeIdx!=-1) {
|
|
MEdge *origMed = &medge[edgeIdx];
|
|
|
|
if (inDLM) {
|
|
flags |= origMed->flag;
|
|
} else {
|
|
flags |= (origMed->flag&ME_SEAM)|ME_EDGEDRAW|ME_EDGERENDER;
|
|
}
|
|
}
|
|
} else {
|
|
flags |= ME_EDGEDRAW | ME_EDGERENDER;
|
|
}
|
|
|
|
for (x=0; x<edgeSize-1; x++) {
|
|
MEdge *med = &dlm->medge[i];
|
|
med->v1 = getEdgeIndex(ss, e, x, edgeSize);
|
|
med->v2 = getEdgeIndex(ss, e, x+1, edgeSize);
|
|
med->flag = flags;
|
|
i++;
|
|
}
|
|
}
|
|
|
|
// load faces
|
|
|
|
i=0;
|
|
for (index=0; index<totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
CCGFace *uvf = faceMap2Uv? faceMap2Uv[index]: NULL;
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
float edge_data[4][6];
|
|
float corner_data[4][6];
|
|
float center_data[6] = {0};
|
|
int numDataComponents = 0;
|
|
TFace *origTFace = NULL;
|
|
int mat_nr;
|
|
int flag;
|
|
|
|
if (!ssFromEditmesh && mface) {
|
|
int origIdx = (int) ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
MFace *origMFace = &mface[origIdx];
|
|
|
|
if (tface) {
|
|
origTFace = &tface[origIdx];
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
unsigned char *col = (unsigned char*) &origTFace->col[S];
|
|
corner_data[S][0] = col[0]/255.0f;
|
|
corner_data[S][1] = col[1]/255.0f;
|
|
corner_data[S][2] = col[2]/255.0f;
|
|
corner_data[S][3] = col[3]/255.0f;
|
|
if (!uvf) {
|
|
corner_data[S][4] = origTFace->uv[S][0];
|
|
corner_data[S][5] = origTFace->uv[S][1];
|
|
}
|
|
}
|
|
numDataComponents = uvf? 4: 6;
|
|
} else if (mcol) {
|
|
MCol *origMCol = &mcol[origIdx*4];
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
unsigned char *col = (unsigned char*) &origMCol[S];
|
|
corner_data[S][0] = col[0]/255.0f;
|
|
corner_data[S][1] = col[1]/255.0f;
|
|
corner_data[S][2] = col[2]/255.0f;
|
|
corner_data[S][3] = col[3]/255.0f;
|
|
}
|
|
numDataComponents = 4;
|
|
}
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
for (k=0; k<numDataComponents; k++) {
|
|
edge_data[S][k] = (corner_data[S][k] + corner_data[(S+1)%numVerts][k])*0.5f;
|
|
center_data[k]+= corner_data[S][k];
|
|
}
|
|
}
|
|
for (k=0; k<numDataComponents; k++) {
|
|
center_data[k]/= numVerts;
|
|
}
|
|
|
|
mat_nr = origMFace->mat_nr;
|
|
flag = origMFace->flag;
|
|
} else if(ssFromEditmesh) {
|
|
EditFace *ef = ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
mat_nr = ef->mat_nr;
|
|
flag = ef->flag;
|
|
} else {
|
|
mat_nr = 0;
|
|
flag = 0;
|
|
}
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
int prevS= (S-1+numVerts)%numVerts;
|
|
|
|
for (y=0; y<gridSize-1; y++) {
|
|
for (x=0; x<gridSize-1; x++) {
|
|
float smoothuv[4][3];
|
|
|
|
MFace *mf = &dlm->mface[i];
|
|
mf->v1 = getFaceIndex(ss, f, S, x+0, y+0, edgeSize, gridSize);
|
|
mf->v2 = getFaceIndex(ss, f, S, x+0, y+1, edgeSize, gridSize);
|
|
mf->v3 = getFaceIndex(ss, f, S, x+1, y+1, edgeSize, gridSize);
|
|
mf->v4 = getFaceIndex(ss, f, S, x+1, y+0, edgeSize, gridSize);
|
|
mf->mat_nr = mat_nr;
|
|
mf->flag = flag;
|
|
|
|
if(uvf) {
|
|
VECCOPY(smoothuv[0], getFaceUV(uvss, uvf, S, x+0, y+0, edgeSize, gridSize));
|
|
VECCOPY(smoothuv[1], getFaceUV(uvss, uvf, S, x+0, y+1, edgeSize, gridSize));
|
|
VECCOPY(smoothuv[2], getFaceUV(uvss, uvf, S, x+1, y+1, edgeSize, gridSize));
|
|
VECCOPY(smoothuv[3], getFaceUV(uvss, uvf, S, x+1, y+0, edgeSize, gridSize));
|
|
}
|
|
|
|
for (j=0; j<4; j++) {
|
|
int fx = x + (j==2||j==3);
|
|
int fy = y + (j==1||j==2);
|
|
float x_v = (float) fx/(gridSize-1);
|
|
float y_v = (float) fy/(gridSize-1);
|
|
float data[6];
|
|
|
|
for (k=0; k<numDataComponents; k++) {
|
|
data[k] = (center_data[k]*(1.0f-x_v) + edge_data[S][k]*x_v)*(1.0f-y_v) +
|
|
(edge_data[prevS][k]*(1.0f-x_v) + corner_data[S][k]*x_v)*y_v;
|
|
}
|
|
|
|
if (dlm->tface) {
|
|
TFace *tf = &dlm->tface[i];
|
|
unsigned char col[4];
|
|
col[0] = (int) (data[0]*255);
|
|
col[1] = (int) (data[1]*255);
|
|
col[2] = (int) (data[2]*255);
|
|
col[3] = (int) (data[3]*255);
|
|
tf->col[j] = *((unsigned int*) col);
|
|
if (uvf) {
|
|
tf->uv[j][0] = smoothuv[j][0];
|
|
tf->uv[j][1] = smoothuv[j][1];
|
|
}
|
|
else {
|
|
tf->uv[j][0] = (float)(data[4]);
|
|
tf->uv[j][1] = (float)(data[5]);
|
|
}
|
|
} else if (dlm->mcol) {
|
|
unsigned char *col = (unsigned char*) &dlm->mcol[i*4+j];
|
|
col[0] = (int) (data[0]*255);
|
|
col[1] = (int) (data[1]*255);
|
|
col[2] = (int) (data[2]*255);
|
|
col[3] = (int) (data[3]*255);
|
|
}
|
|
}
|
|
|
|
if (dlm->tface) {
|
|
TFace *tf = &dlm->tface[i];
|
|
tf->tpage = origTFace->tpage;
|
|
tf->flag = origTFace->flag;
|
|
tf->transp = origTFace->transp;
|
|
tf->mode = origTFace->mode;
|
|
tf->tile = origTFace->tile;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
MEM_freeN(faceMap2);
|
|
MEM_freeN(edgeMap2);
|
|
MEM_freeN(vertMap2);
|
|
|
|
if(uvss) {
|
|
ccgSubSurf_free(uvss);
|
|
MEM_freeN(faceMap2Uv);
|
|
}
|
|
|
|
mesh_calc_normals(dlm->mvert, dlm->totvert, dlm->mface, dlm->totface, &dlm->nors);
|
|
|
|
return dlm;
|
|
}
|
|
|
|
/* face weighting - taken from Brecht's element data patch */
|
|
static void calc_ss_weights(int gridFaces,
|
|
FaceVertWeight **qweight, FaceVertWeight **tweight)
|
|
{
|
|
FaceVertWeight *qw, *tw;
|
|
int x, y, j;
|
|
int numWeights = gridFaces * gridFaces;
|
|
|
|
*tweight = MEM_mallocN(sizeof(**tweight) * numWeights, "ssTriWeight");
|
|
*qweight = MEM_mallocN(sizeof(**qweight) * numWeights, "ssQuadWeight");
|
|
|
|
qw = *qweight;
|
|
tw = *tweight;
|
|
|
|
for (y = 0; y < gridFaces; y++) {
|
|
for (x = 0; x < gridFaces; x++) {
|
|
for (j = 0; j < 4; j++) {
|
|
int fx = x + (j == 2 || j == 3);
|
|
int fy = y + (j == 1 || j == 2);
|
|
float x_v = (float) fx / gridFaces;
|
|
float y_v = (float) fy / gridFaces;
|
|
float tx_v = (1.0f - x_v), ty_v = (1.0f - y_v);
|
|
float center = (1.0f / 3.0f) * tx_v * ty_v;
|
|
|
|
(*tw)[j][0] = center + 0.5f * tx_v * y_v;
|
|
(*tw)[j][2] = center + 0.5f * x_v * ty_v;
|
|
(*tw)[j][1] = 1.0f - (*tw)[j][0] - (*tw)[j][2];
|
|
(*tw)[j][3] = 0.0f;
|
|
|
|
tx_v *= 0.5f;
|
|
ty_v *= 0.5f;
|
|
|
|
(*qw)[j][3] = tx_v * ty_v;
|
|
(*qw)[j][0] = (*qw)[j][3] + tx_v * y_v;
|
|
(*qw)[j][2] = (*qw)[j][3] + x_v * ty_v;
|
|
(*qw)[j][1] = 1.0f - (*qw)[j][0] - (*qw)[j][2] - (*qw)[j][3];
|
|
|
|
}
|
|
tw++;
|
|
qw++;
|
|
}
|
|
}
|
|
}
|
|
|
|
DerivedMesh *ss_to_cdderivedmesh(CCGSubSurf *ss, int ssFromEditmesh,
|
|
int drawInteriorEdges, int useSubsurfUv,
|
|
DerivedMesh *dm)
|
|
{
|
|
DerivedMesh *result;
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int gridFaces = gridSize - 1;
|
|
int edgeBase, faceBase;
|
|
int i, j, k, S, x, y, index;
|
|
int vertBase = 0;
|
|
CCGVertIterator *vi;
|
|
CCGEdgeIterator *ei;
|
|
CCGFaceIterator *fi;
|
|
CCGFace **faceMap2, **faceMap2Uv = NULL;
|
|
CCGEdge **edgeMap2;
|
|
CCGVert **vertMap2;
|
|
int totvert, totedge, totface;
|
|
MVert *mvert;
|
|
MEdge *med;
|
|
MFace *mf;
|
|
TFace *tface;
|
|
CCGSubSurf *uvss = NULL;
|
|
int *origIndex;
|
|
FaceVertWeight *qweight, *tweight;
|
|
|
|
calc_ss_weights(gridFaces, &qweight, &tweight);
|
|
|
|
/* vert map */
|
|
totvert = ccgSubSurf_getNumVerts(ss);
|
|
vertMap2 = MEM_mallocN(totvert*sizeof(*vertMap2), "vertmap");
|
|
vi = ccgSubSurf_getVertIterator(ss);
|
|
for(; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
|
|
if(ssFromEditmesh) {
|
|
vertMap2[ccgDM_getVertMapIndex(NULL, ss, v)] = v;
|
|
} else {
|
|
vertMap2[(int) ccgSubSurf_getVertVertHandle(ss, v)] = v;
|
|
}
|
|
}
|
|
ccgVertIterator_free(vi);
|
|
|
|
totedge = ccgSubSurf_getNumEdges(ss);
|
|
edgeMap2 = MEM_mallocN(totedge*sizeof(*edgeMap2), "edgemap");
|
|
ei = ccgSubSurf_getEdgeIterator(ss);
|
|
for(; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
|
|
if(ssFromEditmesh) {
|
|
edgeMap2[ccgDM_getEdgeMapIndex(NULL, ss, e)] = e;
|
|
} else {
|
|
edgeMap2[(int) ccgSubSurf_getEdgeEdgeHandle(ss, e)] = e;
|
|
}
|
|
}
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
faceMap2 = MEM_mallocN(totface*sizeof(*faceMap2), "facemap");
|
|
fi = ccgSubSurf_getFaceIterator(ss);
|
|
for(; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
|
|
if(ssFromEditmesh) {
|
|
faceMap2[ccgDM_getFaceMapIndex(NULL, ss, f)] = f;
|
|
} else {
|
|
faceMap2[(int) ccgSubSurf_getFaceFaceHandle(ss, f)] = f;
|
|
}
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
|
|
if(dm) {
|
|
result = CDDM_from_template(dm, ccgSubSurf_getNumFinalVerts(ss),
|
|
ccgSubSurf_getNumFinalEdges(ss),
|
|
ccgSubSurf_getNumFinalFaces(ss));
|
|
tface = dm->getFaceDataArray(dm, LAYERTYPE_TFACE);
|
|
} else {
|
|
result = CDDM_new(ccgSubSurf_getNumFinalVerts(ss),
|
|
ccgSubSurf_getNumFinalEdges(ss),
|
|
ccgSubSurf_getNumFinalFaces(ss));
|
|
tface = NULL;
|
|
}
|
|
|
|
if(useSubsurfUv && tface) {
|
|
/* slightly dodgy hack to use current ss_sync_from_uv function */
|
|
DispListMesh dlm;
|
|
|
|
dlm.mface = dm->dupFaceArray(dm);
|
|
dlm.tface = tface;
|
|
dlm.mvert = dm->dupVertArray(dm);
|
|
dlm.totvert = dm->getNumVerts(dm);
|
|
dlm.totface = dm->getNumFaces(dm);
|
|
|
|
/* not for editmesh currently */
|
|
uvss = _getSubSurf(NULL, ccgSubSurf_getSubdivisionLevels(ss),
|
|
0, 1, 0);
|
|
|
|
if(ss_sync_from_uv(uvss, ss, NULL, &dlm)) {
|
|
faceMap2Uv = MEM_mallocN(totface * sizeof(*faceMap2Uv),
|
|
"facemapuv");
|
|
|
|
fi = ccgSubSurf_getFaceIterator(uvss);
|
|
for(; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
faceMap2Uv[(int) ccgSubSurf_getFaceFaceHandle(uvss, f)] = f;
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
|
|
MEM_freeN(dlm.mface);
|
|
MEM_freeN(dlm.mvert);
|
|
}
|
|
|
|
// load verts
|
|
faceBase = i = 0;
|
|
mvert = CDDM_get_verts(result);
|
|
origIndex = result->getVertData(result, 0, LAYERTYPE_ORIGINDEX);
|
|
|
|
for(index = 0; index < totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
FaceVertWeight *weight = (numVerts == 4) ? qweight : tweight;
|
|
int vertIdx[4];
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
|
|
if(ssFromEditmesh)
|
|
vertIdx[S] = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
else
|
|
vertIdx[S] = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
}
|
|
|
|
DM_interp_vert_data(dm, result, vertIdx, weight[0][0], numVerts, i);
|
|
VecCopyf(mvert->co, ccgSubSurf_getFaceCenterData(ss, f));
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++mvert;
|
|
++origIndex;
|
|
i++;
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
int prevS = (S - 1 + numVerts) % numVerts;
|
|
int nextS = (S + 1) % numVerts;
|
|
int otherS = (numVerts == 4) ? (S + 2) % numVerts : 3;
|
|
for(x = 1; x < gridFaces; x++) {
|
|
float w[4];
|
|
w[prevS] = weight[x][0][0];
|
|
w[S] = weight[x][0][1];
|
|
w[nextS] = weight[x][0][2];
|
|
w[otherS] = weight[x][0][3];
|
|
DM_interp_vert_data(dm, result, vertIdx, w, numVerts, i);
|
|
VecCopyf(mvert->co,
|
|
ccgSubSurf_getFaceGridEdgeData(ss, f, S, x));
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++mvert;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
}
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
int prevS = (S - 1 + numVerts) % numVerts;
|
|
int nextS = (S + 1) % numVerts;
|
|
int otherS = (numVerts == 4) ? (S + 2) % numVerts : 3;
|
|
for(y = 1; y < gridFaces; y++) {
|
|
for(x = 1; x < gridFaces; x++) {
|
|
float w[4];
|
|
w[prevS] = weight[y * gridFaces + x][0][0];
|
|
w[S] = weight[y * gridFaces + x][0][1];
|
|
w[nextS] = weight[y * gridFaces + x][0][2];
|
|
w[otherS] = weight[y * gridFaces + x][0][3];
|
|
DM_interp_vert_data(dm, result, vertIdx, w, numVerts, i);
|
|
VecCopyf(mvert->co,
|
|
ccgSubSurf_getFaceGridData(ss, f, S, x, y));
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++mvert;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
*((int*)ccgSubSurf_getFaceUserData(ss, f)) = faceBase;
|
|
faceBase += 1 + numVerts * ((gridSize-2) + (gridSize-2) * (gridSize-2));
|
|
}
|
|
|
|
edgeBase = i;
|
|
for(index = 0; index < totedge; index++) {
|
|
CCGEdge *e = edgeMap2[index];
|
|
int x;
|
|
int vertIdx[2];
|
|
|
|
if(ssFromEditmesh) {
|
|
CCGVert *v;
|
|
v = ccgSubSurf_getEdgeVert0(ss, e);
|
|
vertIdx[0] = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
v = ccgSubSurf_getEdgeVert1(ss, e);
|
|
vertIdx[1] = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
} else {
|
|
CCGVert *v;
|
|
v = ccgSubSurf_getEdgeVert0(ss, e);
|
|
vertIdx[0] = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
v = ccgSubSurf_getEdgeVert1(ss, e);
|
|
vertIdx[1] = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
}
|
|
|
|
for(x = 1; x < edgeSize - 1; x++) {
|
|
float w[2];
|
|
w[1] = (float) x / (edgeSize - 1);
|
|
w[0] = 1 - w[1];
|
|
DM_interp_vert_data(dm, result, vertIdx, w, 2, i);
|
|
VecCopyf(mvert->co, ccgSubSurf_getEdgeData(ss, e, x));
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++mvert;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
|
|
*((int*)ccgSubSurf_getEdgeUserData(ss, e)) = edgeBase;
|
|
edgeBase += edgeSize-2;
|
|
}
|
|
|
|
vertBase = i;
|
|
for(index = 0; index < totvert; index++) {
|
|
CCGVert *v = vertMap2[index];
|
|
int vertIdx;
|
|
|
|
if(ssFromEditmesh)
|
|
vertIdx = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
else
|
|
vertIdx = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
|
|
DM_copy_vert_data(dm, result, vertIdx, i, 1);
|
|
VecCopyf(mvert->co, ccgSubSurf_getVertData(ss, v));
|
|
|
|
*((int*)ccgSubSurf_getVertUserData(ss, v)) = i;
|
|
*origIndex = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
++mvert;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
|
|
// load edges
|
|
i = 0;
|
|
med = CDDM_get_edges(result);
|
|
origIndex = result->getEdgeData(result, 0, LAYERTYPE_ORIGINDEX);
|
|
|
|
for(index = 0; index < totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
for(k = 0; k < numVerts; k++) {
|
|
for(x = 0; x < gridFaces; x++) {
|
|
if(drawInteriorEdges) med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, k, x, 0, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, k, x+1, 0, edgeSize, gridSize);
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++med;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
|
|
for(x = 1; x < gridFaces; x++) {
|
|
for(y = 0; y < gridFaces; y++) {
|
|
if(drawInteriorEdges)
|
|
med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, k, x, y, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, k, x, y + 1,
|
|
edgeSize, gridSize);
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++med;
|
|
++origIndex;
|
|
i++;
|
|
|
|
if(drawInteriorEdges)
|
|
med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, k, y, x, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, k, y + 1, x,
|
|
edgeSize, gridSize);
|
|
*origIndex = ORIGINDEX_NONE;
|
|
++med;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for(index = 0; index < totedge; index++) {
|
|
CCGEdge *e = edgeMap2[index];
|
|
unsigned int flags = 0;
|
|
|
|
if(!ccgSubSurf_getEdgeNumFaces(ss, e)) flags |= ME_LOOSEEDGE;
|
|
|
|
if(ssFromEditmesh) {
|
|
EditEdge *eed = ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
flags |= ME_EDGEDRAW | ME_EDGERENDER;
|
|
if(eed->seam) {
|
|
flags |= ME_SEAM;
|
|
}
|
|
if(eed->sharp) flags |= ME_SHARP;
|
|
} else {
|
|
int edgeIdx = (int)ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
if(edgeIdx != -1 && dm) {
|
|
MEdge origMed;
|
|
dm->getEdge(dm, edgeIdx, &origMed);
|
|
|
|
flags |= origMed.flag;
|
|
}
|
|
}
|
|
|
|
for(x = 0; x < edgeSize - 1; x++) {
|
|
med->v1 = getEdgeIndex(ss, e, x, edgeSize);
|
|
med->v2 = getEdgeIndex(ss, e, x + 1, edgeSize);
|
|
med->flag = flags;
|
|
*origIndex = ccgDM_getEdgeMapIndex(NULL, ss, e);
|
|
++med;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
}
|
|
|
|
// load faces
|
|
i = 0;
|
|
mf = CDDM_get_faces(result);
|
|
origIndex = result->getFaceData(result, 0, LAYERTYPE_ORIGINDEX);
|
|
|
|
for(index = 0; index < totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
CCGFace *uvf = faceMap2Uv ? faceMap2Uv[index] : NULL;
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
int mat_nr;
|
|
int flag;
|
|
int mapIndex = ccgDM_getFaceMapIndex(NULL, ss, f);
|
|
int faceIdx = (int)ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
|
|
if(!ssFromEditmesh) {
|
|
MFace origMFace;
|
|
dm->getFace(dm, faceIdx, &origMFace);
|
|
|
|
mat_nr = origMFace.mat_nr;
|
|
flag = origMFace.flag;
|
|
} else {
|
|
EditFace *ef = ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
mat_nr = ef->mat_nr;
|
|
flag = ef->flag;
|
|
}
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
FaceVertWeight *weight = (numVerts == 4) ? qweight : tweight;
|
|
|
|
for(y = 0; y < gridFaces; y++) {
|
|
for(x = 0; x < gridFaces; x++) {
|
|
mf->v1 = getFaceIndex(ss, f, S, x + 0, y + 0,
|
|
edgeSize, gridSize);
|
|
mf->v2 = getFaceIndex(ss, f, S, x + 0, y + 1,
|
|
edgeSize, gridSize);
|
|
mf->v3 = getFaceIndex(ss, f, S, x + 1, y + 1,
|
|
edgeSize, gridSize);
|
|
mf->v4 = getFaceIndex(ss, f, S, x + 1, y + 0,
|
|
edgeSize, gridSize);
|
|
mf->mat_nr = mat_nr;
|
|
mf->flag = flag;
|
|
|
|
if(dm) {
|
|
int prevS = (S - 1 + numVerts) % numVerts;
|
|
int nextS = (S + 1) % numVerts;
|
|
int otherS = (numVerts == 4) ? (S + 2) % numVerts : 3;
|
|
FaceVertWeight w;
|
|
|
|
for(j = 0; j < 4; ++j) {
|
|
w[j][prevS] = (*weight)[j][0];
|
|
w[j][S] = (*weight)[j][1];
|
|
w[j][nextS] = (*weight)[j][2];
|
|
w[j][otherS] = (*weight)[j][3];
|
|
}
|
|
|
|
DM_interp_face_data(dm, result, &faceIdx, NULL,
|
|
&w, 1, i);
|
|
weight++;
|
|
}
|
|
|
|
if(uvf) {
|
|
TFace *tf = DM_get_face_data(result, i,
|
|
LAYERTYPE_TFACE);
|
|
float *newuv;
|
|
|
|
newuv = getFaceUV(uvss, uvf, S, x + 0, y + 0,
|
|
edgeSize, gridSize);
|
|
tf->uv[0][0] = newuv[0]; tf->uv[0][1] = newuv[1];
|
|
newuv = getFaceUV(uvss, uvf, S, x + 0, y + 1,
|
|
edgeSize, gridSize);
|
|
tf->uv[1][0] = newuv[0]; tf->uv[1][1] = newuv[1];
|
|
newuv = getFaceUV(uvss, uvf, S, x + 1, y + 1,
|
|
edgeSize, gridSize);
|
|
tf->uv[2][0] = newuv[0]; tf->uv[2][1] = newuv[1];
|
|
newuv = getFaceUV(uvss, uvf, S, x + 1, y + 0,
|
|
edgeSize, gridSize);
|
|
tf->uv[3][0] = newuv[0]; tf->uv[3][1] = newuv[1];
|
|
}
|
|
|
|
*origIndex = mapIndex;
|
|
++mf;
|
|
++origIndex;
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
MEM_freeN(faceMap2);
|
|
MEM_freeN(edgeMap2);
|
|
MEM_freeN(vertMap2);
|
|
|
|
MEM_freeN(tweight);
|
|
MEM_freeN(qweight);
|
|
|
|
if(uvss) {
|
|
ccgSubSurf_free(uvss);
|
|
MEM_freeN(faceMap2Uv);
|
|
}
|
|
|
|
CDDM_calc_normals(result);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void ss_sync_from_mesh(CCGSubSurf *ss, Mesh *me, DispListMesh *dlm, float (*vertexCos)[3], int useFlatSubdiv) {
|
|
float creaseFactor = (float) ccgSubSurf_getSubdivisionLevels(ss);
|
|
CCGVertHDL fVerts[4];
|
|
MVert *mvert = dlm?dlm->mvert:me->mvert;
|
|
MEdge *medge = dlm?dlm->medge:me->medge;
|
|
MFace *mface = dlm?dlm->mface:me->mface;
|
|
int totvert = dlm?dlm->totvert:me->totvert;
|
|
int totedge = dlm?dlm->totedge:me->totedge;
|
|
int totface = dlm?dlm->totface:me->totface;
|
|
int i, index;
|
|
|
|
ccgSubSurf_initFullSync(ss);
|
|
|
|
for (i=0,index=-1; i<totvert; i++) {
|
|
CCGVert *v;
|
|
ccgSubSurf_syncVert(ss, (CCGVertHDL) i, vertexCos?vertexCos[i]:mvert[i].co, 0, &v);
|
|
|
|
if (!dlm) index++;
|
|
((int*) ccgSubSurf_getVertUserData(ss, v))[1] = index;
|
|
}
|
|
|
|
if (medge) {
|
|
for (i=0, index=-1; i<totedge; i++) {
|
|
MEdge *med = &medge[i];
|
|
CCGEdge *e;
|
|
float crease = useFlatSubdiv?creaseFactor:med->crease*creaseFactor/255.0f;
|
|
|
|
ccgSubSurf_syncEdge(ss, (CCGEdgeHDL) i, (CCGVertHDL) med->v1, (CCGVertHDL) med->v2, crease, &e);
|
|
|
|
if (!dlm) index++;
|
|
((int*) ccgSubSurf_getEdgeUserData(ss, e))[1] = index;
|
|
}
|
|
}
|
|
|
|
for (i=0, index=-1; i<totface; i++) {
|
|
MFace *mf = &((MFace*) mface)[i];
|
|
CCGFace *f;
|
|
|
|
if (!dlm) index++;
|
|
|
|
fVerts[0] = (CCGVertHDL) mf->v1;
|
|
fVerts[1] = (CCGVertHDL) mf->v2;
|
|
fVerts[2] = (CCGVertHDL) mf->v3;
|
|
fVerts[3] = (CCGVertHDL) mf->v4;
|
|
|
|
// this is very bad, means mesh is internally consistent.
|
|
// it is not really possible to continue without modifying
|
|
// other parts of code significantly to handle missing faces.
|
|
// since this really shouldn't even be possible we just bail.
|
|
if (ccgSubSurf_syncFace(ss, (CCGFaceHDL) i, fVerts[3]?4:3, fVerts, &f)==eCCGError_InvalidValue) {
|
|
static int hasGivenError = 0;
|
|
|
|
if (!hasGivenError) {
|
|
if (!me) {
|
|
error("Unrecoverable error in SubSurf calculation, mesh is inconsistent.");
|
|
} else {
|
|
error("Unrecoverable error in SubSurf calculation, mesh(%s) is inconsistent.", me->id.name+2);
|
|
}
|
|
|
|
hasGivenError = 1;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
((int*) ccgSubSurf_getFaceUserData(ss, f))[1] = index;
|
|
}
|
|
|
|
ccgSubSurf_processSync(ss);
|
|
}
|
|
|
|
static void ss_sync_from_derivedmesh(CCGSubSurf *ss, DerivedMesh *dm,
|
|
float (*vertexCos)[3], int useFlatSubdiv)
|
|
{
|
|
float creaseFactor = (float) ccgSubSurf_getSubdivisionLevels(ss);
|
|
CCGVertHDL fVerts[4];
|
|
int totvert = dm->getNumVerts(dm);
|
|
int totedge = dm->getNumEdges(dm);
|
|
int totface = dm->getNumFaces(dm);
|
|
int i;
|
|
int *index;
|
|
MVert *mvert = dm->dupVertArray(dm);
|
|
MEdge *medge = dm->dupEdgeArray(dm);
|
|
MFace *mface = dm->dupFaceArray(dm);
|
|
MVert *mv;
|
|
MEdge *me;
|
|
MFace *mf;
|
|
|
|
ccgSubSurf_initFullSync(ss);
|
|
|
|
mv = mvert;
|
|
index = (int *)dm->getVertDataArray(dm, LAYERTYPE_ORIGINDEX);
|
|
for(i = 0; i < totvert; i++, mv++, index++) {
|
|
CCGVert *v;
|
|
|
|
if(vertexCos) {
|
|
ccgSubSurf_syncVert(ss, (CCGVertHDL)i, vertexCos[i], 0, &v);
|
|
} else {
|
|
ccgSubSurf_syncVert(ss, (CCGVertHDL)i, mv->co, 0, &v);
|
|
}
|
|
|
|
((int*)ccgSubSurf_getVertUserData(ss, v))[1] = *index;
|
|
}
|
|
|
|
me = medge;
|
|
index = (int *)dm->getEdgeDataArray(dm, LAYERTYPE_ORIGINDEX);
|
|
for(i = 0; i < totedge; i++, me++, index++) {
|
|
CCGEdge *e;
|
|
float crease;
|
|
|
|
crease = useFlatSubdiv ? creaseFactor :
|
|
me->crease * creaseFactor / 255.0f;
|
|
|
|
ccgSubSurf_syncEdge(ss, (CCGEdgeHDL)i, (CCGVertHDL)me->v1,
|
|
(CCGVertHDL)me->v2, crease, &e);
|
|
|
|
((int*)ccgSubSurf_getEdgeUserData(ss, e))[1] = *index;
|
|
}
|
|
|
|
mf = mface;
|
|
index = (int *)dm->getFaceDataArray(dm, LAYERTYPE_ORIGINDEX);
|
|
for (i = 0; i < totface; i++, mf++, index++) {
|
|
CCGFace *f;
|
|
|
|
fVerts[0] = (CCGVertHDL) mf->v1;
|
|
fVerts[1] = (CCGVertHDL) mf->v2;
|
|
fVerts[2] = (CCGVertHDL) mf->v3;
|
|
fVerts[3] = (CCGVertHDL) mf->v4;
|
|
|
|
// this is very bad, means mesh is internally consistent.
|
|
// it is not really possible to continue without modifying
|
|
// other parts of code significantly to handle missing faces.
|
|
// since this really shouldn't even be possible we just bail.
|
|
if(ccgSubSurf_syncFace(ss, (CCGFaceHDL)i, fVerts[3] ? 4 : 3,
|
|
fVerts, &f) == eCCGError_InvalidValue) {
|
|
static int hasGivenError = 0;
|
|
|
|
if(!hasGivenError) {
|
|
error("Unrecoverable error in SubSurf calculation,"
|
|
" mesh is inconsistent.");
|
|
|
|
hasGivenError = 1;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
((int*)ccgSubSurf_getFaceUserData(ss, f))[1] = *index;
|
|
}
|
|
|
|
ccgSubSurf_processSync(ss);
|
|
|
|
MEM_freeN(mvert);
|
|
MEM_freeN(medge);
|
|
MEM_freeN(mface);
|
|
}
|
|
|
|
void ss_sync_from_editmesh(CCGSubSurf *ss, EditMesh *em, float (*vertCos)[3], int useFlatSubdiv)
|
|
{
|
|
float creaseFactor = (float) ccgSubSurf_getSubdivisionLevels(ss);
|
|
EditVert *ev, *fVerts[4];
|
|
EditEdge *ee;
|
|
EditFace *ef;
|
|
int i;
|
|
|
|
ccgSubSurf_initFullSync(ss);
|
|
|
|
if (vertCos) {
|
|
for (i=0,ev=em->verts.first; ev; i++,ev=ev->next) {
|
|
CCGVert *v;
|
|
ccgSubSurf_syncVert(ss, ev, vertCos[i], 0, &v);
|
|
((int*) ccgSubSurf_getVertUserData(ss, v))[1] = i;
|
|
}
|
|
} else {
|
|
for (i=0,ev=em->verts.first; ev; i++,ev=ev->next) {
|
|
CCGVert *v;
|
|
ccgSubSurf_syncVert(ss, ev, ev->co, 0, &v);
|
|
((int*) ccgSubSurf_getVertUserData(ss, v))[1] = i;
|
|
}
|
|
}
|
|
|
|
for (i=0,ee=em->edges.first; ee; i++,ee=ee->next) {
|
|
CCGEdge *e;
|
|
ccgSubSurf_syncEdge(ss, ee, ee->v1, ee->v2, useFlatSubdiv?creaseFactor:ee->crease*creaseFactor, &e);
|
|
((int*) ccgSubSurf_getEdgeUserData(ss, e))[1] = i;
|
|
}
|
|
|
|
for (i=0,ef=em->faces.first; ef; i++,ef=ef->next) {
|
|
CCGFace *f;
|
|
|
|
fVerts[0] = ef->v1;
|
|
fVerts[1] = ef->v2;
|
|
fVerts[2] = ef->v3;
|
|
fVerts[3] = ef->v4;
|
|
|
|
ccgSubSurf_syncFace(ss, ef, ef->v4?4:3, (CCGVertHDL*) fVerts, &f);
|
|
((int*) ccgSubSurf_getFaceUserData(ss, f))[1] = i;
|
|
}
|
|
|
|
ccgSubSurf_processSync(ss);
|
|
}
|
|
|
|
/***/
|
|
|
|
static int ccgDM_getVertMapIndex(CCGDerivedMesh *ccgdm, CCGSubSurf *ss, CCGVert *v) {
|
|
return ((int*) ccgSubSurf_getVertUserData(ss, v))[1];
|
|
}
|
|
|
|
static int ccgDM_getEdgeMapIndex(CCGDerivedMesh *ccgdm, CCGSubSurf *ss, CCGEdge *e) {
|
|
return ((int*) ccgSubSurf_getEdgeUserData(ss, e))[1];
|
|
}
|
|
|
|
static int ccgDM_getFaceMapIndex(CCGDerivedMesh *ccgdm, CCGSubSurf *ss, CCGFace *f) {
|
|
return ((int*) ccgSubSurf_getFaceUserData(ss, f))[1];
|
|
}
|
|
|
|
static void ccgDM_getMinMax(DerivedMesh *dm, float min_r[3], float max_r[3]) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGVertIterator *vi = ccgSubSurf_getVertIterator(ss);
|
|
CCGEdgeIterator *ei = ccgSubSurf_getEdgeIterator(ss);
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
int i, edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
|
|
if (!ccgSubSurf_getNumVerts(ss))
|
|
min_r[0] = min_r[1] = min_r[2] = max_r[0] = max_r[1] = max_r[2] = 0.0;
|
|
|
|
for (; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
float *co = ccgSubSurf_getVertData(ss, v);
|
|
|
|
DO_MINMAX(co, min_r, max_r);
|
|
}
|
|
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
VertData *edgeData = ccgSubSurf_getEdgeDataArray(ss, e);
|
|
|
|
for (i=0; i<edgeSize; i++)
|
|
DO_MINMAX(edgeData[i].co, min_r, max_r);
|
|
}
|
|
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int S, x, y, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
VertData *faceGridData = ccgSubSurf_getFaceGridDataArray(ss, f, S);
|
|
|
|
for (y=0; y<gridSize; y++)
|
|
for (x=0; x<gridSize; x++)
|
|
DO_MINMAX(faceGridData[y*gridSize + x].co, min_r, max_r);
|
|
}
|
|
}
|
|
|
|
ccgFaceIterator_free(fi);
|
|
ccgEdgeIterator_free(ei);
|
|
ccgVertIterator_free(vi);
|
|
}
|
|
static int ccgDM_getNumVerts(DerivedMesh *dm) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
|
|
return ccgSubSurf_getNumFinalVerts(ccgdm->ss);
|
|
}
|
|
static int ccgDM_getNumEdges(DerivedMesh *dm) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
|
|
return ccgSubSurf_getNumFinalEdges(ccgdm->ss);
|
|
}
|
|
static int ccgDM_getNumFaces(DerivedMesh *dm) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
|
|
return ccgSubSurf_getNumFinalFaces(ccgdm->ss);
|
|
}
|
|
|
|
static void ccgDM_getFinalVert(DerivedMesh *dm, int vertNum, MVert *mv)
|
|
{
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int i;
|
|
|
|
memset(mv, 0, sizeof(*mv));
|
|
|
|
if(vertNum < ccgdm->edgeMap[0].startVert) {
|
|
/* this vert comes from face data */
|
|
int lastface = ccgSubSurf_getNumFaces(ss) - 1;
|
|
CCGFace *f;
|
|
int x, y, grid, numVerts;
|
|
int offset;
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int gridSideVerts;
|
|
int gridInternalVerts;
|
|
int gridSideEnd;
|
|
int gridInternalEnd;
|
|
|
|
i = 0;
|
|
while(i < lastface && vertNum >= ccgdm->faceMap[i + 1].startVert)
|
|
++i;
|
|
|
|
f = ccgdm->faceMap[i].face;
|
|
numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
gridSideVerts = gridSize - 2;
|
|
gridInternalVerts = gridSideVerts * gridSideVerts;
|
|
|
|
gridSideEnd = 1 + numVerts * gridSideVerts;
|
|
gridInternalEnd = gridSideEnd + numVerts * gridInternalVerts;
|
|
|
|
offset = vertNum - ccgdm->faceMap[i].startVert;
|
|
if(offset < 1) {
|
|
VecCopyf(mv->co, ccgSubSurf_getFaceCenterData(ss, f));
|
|
} else if(offset < gridSideEnd) {
|
|
offset -= 1;
|
|
grid = offset / gridSideVerts;
|
|
x = offset % gridSideVerts + 1;
|
|
VecCopyf(mv->co, ccgSubSurf_getFaceGridEdgeData(ss, f, grid, x));
|
|
} else if(offset < gridInternalEnd) {
|
|
offset -= gridSideEnd;
|
|
grid = offset / gridInternalVerts;
|
|
offset %= gridInternalVerts;
|
|
y = offset / gridSideVerts + 1;
|
|
x = offset % gridSideVerts + 1;
|
|
VecCopyf(mv->co, ccgSubSurf_getFaceGridData(ss, f, grid, x, y));
|
|
}
|
|
} else if(vertNum < ccgdm->vertMap[0].startVert) {
|
|
/* this vert comes from edge data */
|
|
CCGEdge *e;
|
|
int lastedge = ccgSubSurf_getNumEdges(ss) - 1;
|
|
int x;
|
|
|
|
i = 0;
|
|
while(i < lastedge && vertNum >= ccgdm->edgeMap[i + 1].startVert)
|
|
++i;
|
|
|
|
e = ccgdm->edgeMap[i].edge;
|
|
|
|
x = vertNum - ccgdm->edgeMap[i].startVert + 1;
|
|
VecCopyf(mv->co, ccgSubSurf_getEdgeData(ss, e, x));
|
|
} else {
|
|
/* this vert comes from vert data */
|
|
CCGVert *v;
|
|
i = vertNum - ccgdm->vertMap[0].startVert;
|
|
|
|
v = ccgdm->vertMap[i].vert;
|
|
VecCopyf(mv->co, ccgSubSurf_getVertData(ss, v));
|
|
}
|
|
}
|
|
|
|
static void ccgDM_getFinalEdge(DerivedMesh *dm, int edgeNum, MEdge *med)
|
|
{
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int i;
|
|
|
|
memset(med, 0, sizeof(*med));
|
|
|
|
if(edgeNum < ccgdm->edgeMap[0].startEdge) {
|
|
/* this edge comes from face data */
|
|
int lastface = ccgSubSurf_getNumFaces(ss) - 1;
|
|
CCGFace *f;
|
|
int x, y, grid, numVerts;
|
|
int offset;
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSideEdges;
|
|
int gridInternalEdges;
|
|
|
|
i = 0;
|
|
while(i < lastface && edgeNum >= ccgdm->faceMap[i + 1].startEdge)
|
|
++i;
|
|
|
|
f = ccgdm->faceMap[i].face;
|
|
numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
gridSideEdges = gridSize - 1;
|
|
gridInternalEdges = (gridSideEdges - 1) * gridSideEdges * 2;
|
|
|
|
offset = edgeNum - ccgdm->faceMap[i].startEdge;
|
|
grid = offset / (gridSideEdges + gridInternalEdges);
|
|
offset %= (gridSideEdges + gridInternalEdges);
|
|
|
|
if(offset < gridSideEdges) {
|
|
x = offset;
|
|
med->v1 = getFaceIndex(ss, f, grid, x, 0, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, grid, x+1, 0, edgeSize, gridSize);
|
|
} else {
|
|
offset -= gridSideEdges;
|
|
x = (offset / 2) / gridSideEdges + 1;
|
|
y = (offset / 2) % gridSideEdges;
|
|
if(offset % 2 == 0) {
|
|
med->v1 = getFaceIndex(ss, f, grid, x, y, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, grid, x, y+1, edgeSize, gridSize);
|
|
} else {
|
|
med->v1 = getFaceIndex(ss, f, grid, y, x, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, grid, y+1, x, edgeSize, gridSize);
|
|
}
|
|
}
|
|
} else {
|
|
/* this vert comes from edge data */
|
|
CCGEdge *e;
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int x;
|
|
unsigned int flags = 0;
|
|
|
|
i = (edgeNum - ccgdm->edgeMap[0].startEdge) / (edgeSize - 1);
|
|
|
|
e = ccgdm->edgeMap[i].edge;
|
|
|
|
if(!ccgSubSurf_getEdgeNumFaces(ss, e)) flags |= ME_LOOSEEDGE;
|
|
|
|
x = edgeNum - ccgdm->edgeMap[i].startEdge;
|
|
|
|
med->v1 = getEdgeIndex(ss, e, x, edgeSize);
|
|
med->v2 = getEdgeIndex(ss, e, x+1, edgeSize);
|
|
|
|
if(ccgdm->fromEditmesh) {
|
|
EditEdge *eed = ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
flags |= ME_EDGEDRAW | ME_EDGERENDER;
|
|
if(eed->seam) {
|
|
flags |= ME_SEAM;
|
|
}
|
|
if(eed->sharp) flags |= ME_SHARP;
|
|
} else if(ccgdm->dlm || ccgdm->me) {
|
|
int edgeIdx = (int) ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
if(edgeIdx!=-1) {
|
|
MEdge *medge = (ccgdm->dlm ? ccgdm->dlm->medge
|
|
: ccgdm->me->medge);
|
|
MEdge *origMed = &medge[edgeIdx];
|
|
|
|
if(ccgdm->dlm) {
|
|
flags |= origMed->flag;
|
|
} else {
|
|
flags |= (origMed->flag & (ME_SEAM | ME_SHARP))
|
|
| ME_EDGEDRAW | ME_EDGERENDER;
|
|
}
|
|
}
|
|
} else {
|
|
flags |= ME_EDGEDRAW | ME_EDGERENDER;
|
|
}
|
|
|
|
med->flag = flags;
|
|
}
|
|
}
|
|
|
|
static void ccgDM_getFinalFace(DerivedMesh *dm, int faceNum, MFace *mf)
|
|
{
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSideEdges = gridSize - 1;
|
|
int gridFaces = gridSideEdges * gridSideEdges;
|
|
int i;
|
|
CCGFace *f;
|
|
int numVerts;
|
|
int offset;
|
|
int grid;
|
|
int x, y;
|
|
int lastface = ccgSubSurf_getNumFaces(ss) - 1;
|
|
int *faceFlags = dm->getFaceData(dm, faceNum, LAYERTYPE_FLAGS);
|
|
|
|
memset(mf, 0, sizeof(*mf));
|
|
|
|
i = 0;
|
|
while(i < lastface && faceNum >= ccgdm->faceMap[i + 1].startFace)
|
|
++i;
|
|
|
|
f = ccgdm->faceMap[i].face;
|
|
numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
offset = faceNum - ccgdm->faceMap[i].startFace;
|
|
grid = offset / gridFaces;
|
|
offset %= gridFaces;
|
|
y = offset / gridSideEdges;
|
|
x = offset % gridSideEdges;
|
|
|
|
mf->v1 = getFaceIndex(ss, f, grid, x+0, y+0, edgeSize, gridSize);
|
|
mf->v2 = getFaceIndex(ss, f, grid, x+0, y+1, edgeSize, gridSize);
|
|
mf->v3 = getFaceIndex(ss, f, grid, x+1, y+1, edgeSize, gridSize);
|
|
mf->v4 = getFaceIndex(ss, f, grid, x+1, y+0, edgeSize, gridSize);
|
|
|
|
if(faceFlags) mf->flag = *faceFlags;
|
|
else mf->flag = ME_SMOOTH;
|
|
}
|
|
|
|
static void ccgDM_getFinalVertArray(DerivedMesh *dm, MVert *mvert)
|
|
{
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int index;
|
|
int totvert, totedge, totface;
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int i = 0;
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
for(index = 0; index < totface; index++) {
|
|
CCGFace *f = ccgdm->faceMap[index].face;
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
VecCopyf(mvert[i++].co, ccgSubSurf_getFaceCenterData(ss, f));
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
for(x = 1; x < gridSize - 1; x++) {
|
|
VecCopyf(mvert[i++].co,
|
|
ccgSubSurf_getFaceGridEdgeData(ss, f, S, x));
|
|
}
|
|
}
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
for(y = 1; y < gridSize - 1; y++) {
|
|
for(x = 1; x < gridSize - 1; x++) {
|
|
VecCopyf(mvert[i++].co,
|
|
ccgSubSurf_getFaceGridData(ss, f, S, x, y));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
totedge = ccgSubSurf_getNumEdges(ss);
|
|
for(index = 0; index < totedge; index++) {
|
|
CCGEdge *e = ccgdm->edgeMap[index].edge;
|
|
int x;
|
|
|
|
for(x = 1; x < edgeSize - 1; x++) {
|
|
VecCopyf(mvert[i++].co, ccgSubSurf_getEdgeData(ss, e, x));
|
|
}
|
|
}
|
|
|
|
totvert = ccgSubSurf_getNumVerts(ss);
|
|
for(index = 0; index < totvert; index++) {
|
|
CCGVert *v = ccgdm->vertMap[index].vert;
|
|
|
|
VecCopyf(mvert[i].co, ccgSubSurf_getVertData(ss, v));
|
|
|
|
i++;
|
|
}
|
|
}
|
|
|
|
static void ccgDM_getFinalEdgeArray(DerivedMesh *dm, MEdge *medge)
|
|
{
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int index;
|
|
int totedge, totface;
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int i = 0;
|
|
MEdge *origEdges = NULL;
|
|
|
|
if(!ccgdm->fromEditmesh) {
|
|
if(ccgdm->dlm) origEdges = ccgdm->dlm->medge;
|
|
else if(ccgdm->me) origEdges = ccgdm->me->medge;
|
|
}
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
for(index = 0; index < totface; index++) {
|
|
CCGFace *f = ccgdm->faceMap[index].face;
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
for(x = 0; x < gridSize - 1; x++) {
|
|
MEdge *med = &medge[i];
|
|
|
|
if(ccgdm->drawInteriorEdges)
|
|
med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, S, x, 0, edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, S, x + 1, 0, edgeSize, gridSize);
|
|
i++;
|
|
}
|
|
|
|
for(x = 1; x < gridSize - 1; x++) {
|
|
for(y = 0; y < gridSize - 1; y++) {
|
|
MEdge *med;
|
|
|
|
med = &medge[i];
|
|
if(ccgdm->drawInteriorEdges)
|
|
med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, S, x, y,
|
|
edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, S, x, y + 1,
|
|
edgeSize, gridSize);
|
|
i++;
|
|
|
|
med = &medge[i];
|
|
if(ccgdm->drawInteriorEdges)
|
|
med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
med->v1 = getFaceIndex(ss, f, S, y, x,
|
|
edgeSize, gridSize);
|
|
med->v2 = getFaceIndex(ss, f, S, y + 1, x,
|
|
edgeSize, gridSize);
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
totedge = ccgSubSurf_getNumEdges(ss);
|
|
for(index = 0; index < totedge; index++) {
|
|
CCGEdge *e = ccgdm->edgeMap[index].edge;
|
|
unsigned int flags = 0;
|
|
int x;
|
|
|
|
if(!ccgSubSurf_getEdgeNumFaces(ss, e)) flags |= ME_LOOSEEDGE;
|
|
|
|
if(ccgdm->fromEditmesh) {
|
|
EditEdge *eed = ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
flags |= ME_EDGEDRAW | ME_EDGERENDER;
|
|
if (eed->seam) {
|
|
flags |= ME_SEAM;
|
|
}
|
|
if(eed->sharp) flags |= ME_SHARP;
|
|
} else if(origEdges){
|
|
int edgeIdx = (int)ccgSubSurf_getEdgeEdgeHandle(ss, e);
|
|
|
|
if(edgeIdx != -1) {
|
|
MEdge *origMed = &origEdges[edgeIdx];
|
|
|
|
if(ccgdm->dlm) {
|
|
flags |= origMed->flag;
|
|
} else {
|
|
flags |= (origMed->flag & (ME_SEAM | ME_SHARP))
|
|
| ME_EDGEDRAW | ME_EDGERENDER;
|
|
}
|
|
}
|
|
} else {
|
|
flags |= ME_EDGEDRAW | ME_EDGERENDER;
|
|
}
|
|
|
|
for(x = 0; x < edgeSize - 1; x++) {
|
|
MEdge *med = &medge[i];
|
|
med->v1 = getEdgeIndex(ss, e, x, edgeSize);
|
|
med->v2 = getEdgeIndex(ss, e, x + 1, edgeSize);
|
|
med->flag = flags;
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void ccgDM_getFinalFaceArray(DerivedMesh *dm, MFace *mface)
|
|
{
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int index;
|
|
int totface;
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int i = 0;
|
|
MFace *origFaces = NULL;
|
|
int *faceFlags = dm->getFaceDataArray(dm, LAYERTYPE_FLAGS);
|
|
|
|
if(!ccgdm->fromEditmesh) {
|
|
if(ccgdm->dlm) origFaces = ccgdm->dlm->mface;
|
|
else if(ccgdm->me) origFaces = ccgdm->me->mface;
|
|
}
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
for(index = 0; index < totface; index++) {
|
|
CCGFace *f = ccgdm->faceMap[index].face;
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
int mat_nr = 0;
|
|
int flag = ME_SMOOTH; /* assume face is smooth by default */
|
|
|
|
if(!faceFlags) {
|
|
if(!ccgdm->fromEditmesh && origFaces) {
|
|
int origIdx = (int) ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
MFace *origMFace = &origFaces[origIdx];
|
|
|
|
mat_nr = origMFace->mat_nr;
|
|
flag = origMFace->flag;
|
|
} else if(ccgdm->fromEditmesh) {
|
|
EditFace *ef = ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
mat_nr = ef->mat_nr;
|
|
flag = ef->flag;
|
|
}
|
|
}
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
for(y = 0; y < gridSize - 1; y++) {
|
|
for(x = 0; x < gridSize - 1; x++) {
|
|
MFace *mf = &mface[i];
|
|
mf->v1 = getFaceIndex(ss, f, S, x + 0, y + 0,
|
|
edgeSize, gridSize);
|
|
mf->v2 = getFaceIndex(ss, f, S, x + 0, y + 1,
|
|
edgeSize, gridSize);
|
|
mf->v3 = getFaceIndex(ss, f, S, x + 1, y + 1,
|
|
edgeSize, gridSize);
|
|
mf->v4 = getFaceIndex(ss, f, S, x + 1, y + 0,
|
|
edgeSize, gridSize);
|
|
mf->mat_nr = mat_nr;
|
|
if(faceFlags) mf->flag = faceFlags[i];
|
|
else mf->flag = flag;
|
|
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void ccgdm_getVertCos(DerivedMesh *dm, float (*cos)[3]) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int i;
|
|
CCGVertIterator *vi;
|
|
CCGEdgeIterator *ei;
|
|
CCGFaceIterator *fi;
|
|
CCGFace **faceMap2;
|
|
CCGEdge **edgeMap2;
|
|
CCGVert **vertMap2;
|
|
int index, totvert, totedge, totface;
|
|
|
|
totvert = ccgSubSurf_getNumVerts(ss);
|
|
vertMap2 = MEM_mallocN(totvert*sizeof(*vertMap2), "vertmap");
|
|
vi = ccgSubSurf_getVertIterator(ss);
|
|
for (; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
|
|
if (ccgdm->fromEditmesh) {
|
|
vertMap2[ccgDM_getVertMapIndex(ccgdm,ss,v)] = v;
|
|
} else {
|
|
vertMap2[(int) ccgSubSurf_getVertVertHandle(ss, v)] = v;
|
|
}
|
|
}
|
|
ccgVertIterator_free(vi);
|
|
|
|
totedge = ccgSubSurf_getNumEdges(ss);
|
|
edgeMap2 = MEM_mallocN(totedge*sizeof(*edgeMap2), "edgemap");
|
|
ei = ccgSubSurf_getEdgeIterator(ss);
|
|
for (i=0; !ccgEdgeIterator_isStopped(ei); i++,ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
|
|
if (ccgdm->fromEditmesh) {
|
|
edgeMap2[ccgDM_getEdgeMapIndex(ccgdm,ss,e)] = e;
|
|
} else {
|
|
edgeMap2[(int) ccgSubSurf_getEdgeEdgeHandle(ss, e)] = e;
|
|
}
|
|
}
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
faceMap2 = MEM_mallocN(totface*sizeof(*faceMap2), "facemap");
|
|
fi = ccgSubSurf_getFaceIterator(ss);
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
|
|
if (ccgdm->fromEditmesh) {
|
|
faceMap2[ccgDM_getFaceMapIndex(ccgdm,ss,f)] = f;
|
|
} else {
|
|
faceMap2[(int) ccgSubSurf_getFaceFaceHandle(ss, f)] = f;
|
|
}
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
|
|
i = 0;
|
|
for (index=0; index<totface; index++) {
|
|
CCGFace *f = faceMap2[index];
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
VecCopyf(cos[i++], ccgSubSurf_getFaceCenterData(ss, f));
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
for (x=1; x<gridSize-1; x++) {
|
|
VecCopyf(cos[i++], ccgSubSurf_getFaceGridEdgeData(ss, f, S, x));
|
|
}
|
|
}
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
for (y=1; y<gridSize-1; y++) {
|
|
for (x=1; x<gridSize-1; x++) {
|
|
VecCopyf(cos[i++], ccgSubSurf_getFaceGridData(ss, f, S, x, y));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (index=0; index<totedge; index++) {
|
|
CCGEdge *e= edgeMap2[index];
|
|
int x;
|
|
|
|
for (x=1; x<edgeSize-1; x++) {
|
|
VecCopyf(cos[i++], ccgSubSurf_getEdgeData(ss, e, x));
|
|
}
|
|
}
|
|
|
|
for (index=0; index<totvert; index++) {
|
|
CCGVert *v = vertMap2[index];
|
|
VecCopyf(cos[i++], ccgSubSurf_getVertData(ss, v));
|
|
}
|
|
|
|
MEM_freeN(vertMap2);
|
|
MEM_freeN(edgeMap2);
|
|
MEM_freeN(faceMap2);
|
|
}
|
|
static void ccgDM_foreachMappedVert(DerivedMesh *dm, void (*func)(void *userData, int index, float *co, float *no_f, short *no_s), void *userData) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGVertIterator *vi = ccgSubSurf_getVertIterator(ccgdm->ss);
|
|
|
|
for (; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
VertData *vd = ccgSubSurf_getVertData(ccgdm->ss, v);
|
|
int index = ccgDM_getVertMapIndex(ccgdm, ccgdm->ss, v);
|
|
|
|
if (index!=-1)
|
|
func(userData, index, vd->co, vd->no, NULL);
|
|
}
|
|
|
|
ccgVertIterator_free(vi);
|
|
}
|
|
static void ccgDM_foreachMappedEdge(DerivedMesh *dm, void (*func)(void *userData, int index, float *v0co, float *v1co), void *userData) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGEdgeIterator *ei = ccgSubSurf_getEdgeIterator(ss);
|
|
int i, edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
VertData *edgeData = ccgSubSurf_getEdgeDataArray(ss, e);
|
|
int index = ccgDM_getEdgeMapIndex(ccgdm, ss, e);
|
|
|
|
if (index!=-1) {
|
|
for (i=0; i<edgeSize-1; i++)
|
|
func(userData, index, edgeData[i].co, edgeData[i+1].co);
|
|
}
|
|
}
|
|
|
|
ccgEdgeIterator_free(ei);
|
|
}
|
|
static DispListMesh *ccgDM_convertToDispListMesh(DerivedMesh *dm, int allowShared) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
|
|
return ss_to_displistmesh(ccgdm->ss, ccgdm, ccgdm->fromEditmesh, ccgdm->drawInteriorEdges, ccgdm->useSubsurfUv, ccgdm->me, ccgdm->dlm);
|
|
}
|
|
|
|
static void ccgDM_drawVerts(DerivedMesh *dm) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
int edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
CCGVertIterator *vi;
|
|
CCGEdgeIterator *ei;
|
|
CCGFaceIterator *fi;
|
|
|
|
glBegin(GL_POINTS);
|
|
vi = ccgSubSurf_getVertIterator(ss);
|
|
for (; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
glVertex3fv(ccgSubSurf_getVertData(ss, v));
|
|
}
|
|
ccgVertIterator_free(vi);
|
|
|
|
ei = ccgSubSurf_getEdgeIterator(ss);
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
int x;
|
|
|
|
for (x=1; x<edgeSize-1; x++)
|
|
glVertex3fv(ccgSubSurf_getEdgeData(ss, e, x));
|
|
}
|
|
ccgEdgeIterator_free(ei);
|
|
|
|
fi = ccgSubSurf_getFaceIterator(ss);
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int x, y, S, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
glVertex3fv(ccgSubSurf_getFaceCenterData(ss, f));
|
|
for (S=0; S<numVerts; S++)
|
|
for (x=1; x<gridSize-1; x++)
|
|
glVertex3fv(ccgSubSurf_getFaceGridEdgeData(ss, f, S, x));
|
|
for (S=0; S<numVerts; S++)
|
|
for (y=1; y<gridSize-1; y++)
|
|
for (x=1; x<gridSize-1; x++)
|
|
glVertex3fv(ccgSubSurf_getFaceGridData(ss, f, S, x, y));
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
glEnd();
|
|
}
|
|
static void ccgDM_drawEdges(DerivedMesh *dm, int drawLooseEdges) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGEdgeIterator *ei = ccgSubSurf_getEdgeIterator(ss);
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
int i, edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
int useAging;
|
|
|
|
ccgSubSurf_getUseAgeCounts(ss, &useAging, NULL, NULL, NULL);
|
|
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
VertData *edgeData = ccgSubSurf_getEdgeDataArray(ss, e);
|
|
|
|
if (!drawLooseEdges && !ccgSubSurf_getEdgeNumFaces(ss, e))
|
|
continue;
|
|
|
|
if (useAging && !(G.f&G_BACKBUFSEL)) {
|
|
int ageCol = 255-ccgSubSurf_getEdgeAge(ss, e)*4;
|
|
glColor3ub(0, ageCol>0?ageCol:0, 0);
|
|
}
|
|
|
|
glBegin(GL_LINE_STRIP);
|
|
for (i=0; i<edgeSize-1; i++) {
|
|
glVertex3fv(edgeData[i].co);
|
|
glVertex3fv(edgeData[i+1].co);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
if (useAging && !(G.f&G_BACKBUFSEL)) {
|
|
glColor3ub(0, 0, 0);
|
|
}
|
|
|
|
if (ccgdm->drawInteriorEdges) {
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int S, x, y, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
VertData *faceGridData = ccgSubSurf_getFaceGridDataArray(ss, f, S);
|
|
|
|
glBegin(GL_LINE_STRIP);
|
|
for (x=0; x<gridSize; x++)
|
|
glVertex3fv(faceGridData[x].co);
|
|
glEnd();
|
|
for (y=1; y<gridSize-1; y++) {
|
|
glBegin(GL_LINE_STRIP);
|
|
for (x=0; x<gridSize; x++)
|
|
glVertex3fv(faceGridData[y*gridSize + x].co);
|
|
glEnd();
|
|
}
|
|
for (x=1; x<gridSize-1; x++) {
|
|
glBegin(GL_LINE_STRIP);
|
|
for (y=0; y<gridSize; y++)
|
|
glVertex3fv(faceGridData[y*gridSize + x].co);
|
|
glEnd();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
ccgFaceIterator_free(fi);
|
|
ccgEdgeIterator_free(ei);
|
|
}
|
|
static void ccgDM_drawLooseEdges(DerivedMesh *dm) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGEdgeIterator *ei = ccgSubSurf_getEdgeIterator(ss);
|
|
int i, edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
VertData *edgeData = ccgSubSurf_getEdgeDataArray(ss, e);
|
|
|
|
if (!ccgSubSurf_getEdgeNumFaces(ss, e)) {
|
|
glBegin(GL_LINE_STRIP);
|
|
for (i=0; i<edgeSize-1; i++) {
|
|
glVertex3fv(edgeData[i].co);
|
|
glVertex3fv(edgeData[i+1].co);
|
|
}
|
|
glEnd();
|
|
}
|
|
}
|
|
|
|
ccgEdgeIterator_free(ei);
|
|
}
|
|
|
|
/* Only used by non-editmesh types */
|
|
static void ccgDM_drawFacesSolid(DerivedMesh *dm, int (*setMaterial)(int)) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
MFace *mface = ccgdm->dlm?ccgdm->dlm->mface:ccgdm->me->mface;
|
|
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int S, x, y, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
int index = (int) ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
MFace *mf = &mface[index];
|
|
|
|
if (!setMaterial(mf->mat_nr+1))
|
|
continue;
|
|
|
|
glShadeModel((mf->flag&ME_SMOOTH)?GL_SMOOTH:GL_FLAT);
|
|
for (S=0; S<numVerts; S++) {
|
|
VertData *faceGridData = ccgSubSurf_getFaceGridDataArray(ss, f, S);
|
|
|
|
if (mf->flag&ME_SMOOTH) {
|
|
for (y=0; y<gridSize-1; y++) {
|
|
glBegin(GL_QUAD_STRIP);
|
|
for (x=0; x<gridSize; x++) {
|
|
VertData *a = &faceGridData[(y+0)*gridSize + x];
|
|
VertData *b = &faceGridData[(y+1)*gridSize + x];
|
|
|
|
glNormal3fv(a->no);
|
|
glVertex3fv(a->co);
|
|
glNormal3fv(b->no);
|
|
glVertex3fv(b->co);
|
|
}
|
|
glEnd();
|
|
}
|
|
} else {
|
|
glBegin(GL_QUADS);
|
|
for (y=0; y<gridSize-1; y++) {
|
|
for (x=0; x<gridSize-1; x++) {
|
|
float *a = faceGridData[(y+0)*gridSize + x].co;
|
|
float *b = faceGridData[(y+0)*gridSize + x + 1].co;
|
|
float *c = faceGridData[(y+1)*gridSize + x + 1].co;
|
|
float *d = faceGridData[(y+1)*gridSize + x].co;
|
|
float a_cX = c[0]-a[0], a_cY = c[1]-a[1], a_cZ = c[2]-a[2];
|
|
float b_dX = d[0]-b[0], b_dY = d[1]-b[1], b_dZ = d[2]-b[2];
|
|
float no[3];
|
|
|
|
no[0] = b_dY*a_cZ - b_dZ*a_cY;
|
|
no[1] = b_dZ*a_cX - b_dX*a_cZ;
|
|
no[2] = b_dX*a_cY - b_dY*a_cX;
|
|
glNormal3fv(no);
|
|
|
|
glVertex3fv(d);
|
|
glVertex3fv(c);
|
|
glVertex3fv(b);
|
|
glVertex3fv(a);
|
|
}
|
|
}
|
|
glEnd();
|
|
}
|
|
}
|
|
}
|
|
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
static void ccgDM_drawFacesColored(DerivedMesh *dm, int useTwoSided, unsigned char *col1, unsigned char *col2) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
unsigned char *cp1, *cp2;
|
|
int useTwoSide=1;
|
|
|
|
cp1= col1;
|
|
if(col2) {
|
|
cp2= col2;
|
|
} else {
|
|
cp2= NULL;
|
|
useTwoSide= 0;
|
|
}
|
|
|
|
glShadeModel(GL_SMOOTH);
|
|
if(col1 && col2)
|
|
glEnable(GL_CULL_FACE);
|
|
|
|
glBegin(GL_QUADS);
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int S, x, y, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
VertData *faceGridData = ccgSubSurf_getFaceGridDataArray(ss, f, S);
|
|
for (y=0; y<gridSize-1; y++) {
|
|
for (x=0; x<gridSize-1; x++) {
|
|
float *a = faceGridData[(y+0)*gridSize + x].co;
|
|
float *b = faceGridData[(y+0)*gridSize + x + 1].co;
|
|
float *c = faceGridData[(y+1)*gridSize + x + 1].co;
|
|
float *d = faceGridData[(y+1)*gridSize + x].co;
|
|
|
|
glColor3ub(cp1[3], cp1[2], cp1[1]);
|
|
glVertex3fv(d);
|
|
glColor3ub(cp1[7], cp1[6], cp1[5]);
|
|
glVertex3fv(c);
|
|
glColor3ub(cp1[11], cp1[10], cp1[9]);
|
|
glVertex3fv(b);
|
|
glColor3ub(cp1[15], cp1[14], cp1[13]);
|
|
glVertex3fv(a);
|
|
|
|
if (useTwoSide) {
|
|
glColor3ub(cp2[15], cp2[14], cp2[13]);
|
|
glVertex3fv(a);
|
|
glColor3ub(cp2[11], cp2[10], cp2[9]);
|
|
glVertex3fv(b);
|
|
glColor3ub(cp2[7], cp2[6], cp2[5]);
|
|
glVertex3fv(c);
|
|
glColor3ub(cp2[3], cp2[2], cp2[1]);
|
|
glVertex3fv(d);
|
|
}
|
|
|
|
if (cp2) cp2+=16;
|
|
cp1+=16;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
glEnd();
|
|
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
|
|
static void ccgDM_drawFacesTex(DerivedMesh *dm, int (*setDrawParams)(TFace *tface, int matnr))
|
|
{
|
|
/* unimplemented, no textures in editmode anyway */
|
|
}
|
|
static void ccgDM_drawMappedFacesTex(DerivedMesh *dm, int (*setDrawParams)(void *userData, int index), void *userData)
|
|
{
|
|
/* unfinished code, no textures in editmode anyway */
|
|
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
int gridSize = ccgSubSurf_getGridSize(ss);
|
|
MFace *mface = ccgdm->dlm?ccgdm->dlm->mface:ccgdm->me->mface;
|
|
TFace *tface = ccgdm->dlm?ccgdm->dlm->tface:ccgdm->me->tface;
|
|
MCol *mcol = ccgdm->dlm?ccgdm->dlm->mcol:ccgdm->me->mcol;
|
|
// float uv[4][2];
|
|
// float col[4][3];
|
|
|
|
glBegin(GL_QUADS);
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int S, x, y, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
int index = (int) ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
MFace *mf = &mface[index];
|
|
TFace *tf = tface?&tface[index]:NULL;
|
|
unsigned char *cp= NULL;
|
|
int findex = ccgDM_getFaceMapIndex(ccgdm, ss, f);
|
|
int flag = (findex == -1)? 0: setDrawParams(userData, findex);
|
|
|
|
if (flag==0) {
|
|
continue;
|
|
} else if (flag==1) {
|
|
if (tf) {
|
|
cp= (unsigned char*) tf->col;
|
|
} else if (mcol) {
|
|
cp= (unsigned char*) &mcol[index*4];
|
|
}
|
|
}
|
|
|
|
for (S=0; S<numVerts; S++) {
|
|
VertData *faceGridData = ccgSubSurf_getFaceGridDataArray(ss, f, S);
|
|
for (y=0; y<gridSize-1; y++) {
|
|
for (x=0; x<gridSize-1; x++) {
|
|
VertData *a = &faceGridData[(y+0)*gridSize + x + 0];
|
|
VertData *b = &faceGridData[(y+0)*gridSize + x + 1];
|
|
VertData *c = &faceGridData[(y+1)*gridSize + x + 1];
|
|
VertData *d = &faceGridData[(y+1)*gridSize + x + 0];
|
|
|
|
if (!(mf->flag&ME_SMOOTH)) {
|
|
float a_cX = c->co[0]-a->co[0], a_cY = c->co[1]-a->co[1], a_cZ = c->co[2]-a->co[2];
|
|
float b_dX = d->co[0]-b->co[0], b_dY = d->co[1]-b->co[1], b_dZ = d->co[2]-b->co[2];
|
|
float no[3];
|
|
|
|
no[0] = b_dY*a_cZ - b_dZ*a_cY;
|
|
no[1] = b_dZ*a_cX - b_dX*a_cZ;
|
|
no[2] = b_dX*a_cY - b_dY*a_cX;
|
|
|
|
glNormal3fv(no);
|
|
}
|
|
|
|
// if (tf) glTexCoord2fv(tf->uv[0]);
|
|
// if (cp) glColor3ub(cp[3], cp[2], cp[1]);
|
|
// if (mf->flag&ME_SMOOTH) glNormal3sv(mvert[mf->v1].no);
|
|
// glVertex3fv(mvert[mf->v1].co);
|
|
|
|
/*
|
|
{
|
|
float x_v = (float) fx/(gridSize-1);
|
|
float y_v = (float) fy/(gridSize-1);
|
|
float data[6];
|
|
|
|
for (k=0; k<numDataComponents; k++) {
|
|
data[k] = (center_data[k]*(1.0f-x_v) + edge_data[S][k]*x_v)*(1.0f-y_v) +
|
|
(edge_data[prevS][k]*(1.0f-x_v) + corner_data[S][k]*x_v)*y_v;
|
|
}
|
|
*/
|
|
|
|
// if (cp) glColor3ub(cp[3], cp[2], cp[1]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3fv(d->no);
|
|
glVertex3fv(d->co);
|
|
// if (cp) glColor3ub(cp[7], cp[6], cp[5]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3fv(c->no);
|
|
glVertex3fv(c->co);
|
|
// if (cp) glColor3ub(cp[11], cp[10], cp[9]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3fv(b->no);
|
|
glVertex3fv(b->co);
|
|
// if (cp) glColor3ub(cp[15], cp[14], cp[13]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3fv(a->no);
|
|
glVertex3fv(a->co);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
glEnd();
|
|
|
|
ccgFaceIterator_free(fi);
|
|
/*
|
|
MeshDerivedMesh *mdm = (MeshDerivedMesh*) dm;
|
|
Mesh *me = mdm->me;
|
|
MVert *mvert= mdm->verts;
|
|
MFace *mface= me->mface;
|
|
TFace *tface = me->tface;
|
|
float *nors = mdm->nors;
|
|
int a;
|
|
|
|
for (a=0; a<me->totface; a++) {
|
|
MFace *mf= &mface[a];
|
|
if (tf) glTexCoord2fv(tf->uv[1]);
|
|
if (cp) glColor3ub(cp[7], cp[6], cp[5]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3sv(mvert[mf->v2].no);
|
|
glVertex3fv(mvert[mf->v2].co);
|
|
|
|
if (tf) glTexCoord2fv(tf->uv[2]);
|
|
if (cp) glColor3ub(cp[11], cp[10], cp[9]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3sv(mvert[mf->v3].no);
|
|
glVertex3fv(mvert[mf->v3].co);
|
|
|
|
if(mf->v4) {
|
|
if (tf) glTexCoord2fv(tf->uv[3]);
|
|
if (cp) glColor3ub(cp[15], cp[14], cp[13]);
|
|
if (mf->flag&ME_SMOOTH) glNormal3sv(mvert[mf->v4].no);
|
|
glVertex3fv(mvert[mf->v4].co);
|
|
}
|
|
glEnd();
|
|
}
|
|
*/
|
|
}
|
|
static void ccgDM_drawMappedFaces(DerivedMesh *dm, int (*setDrawOptions)(void *userData, int index, int *drawSmooth_r), void *userData, int useColors) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
int i, gridSize = ccgSubSurf_getGridSize(ss);
|
|
int *faceFlags = dm->getFaceDataArray(dm, LAYERTYPE_FLAGS);
|
|
|
|
for (i=0; !ccgFaceIterator_isStopped(fi); i++,ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int S, x, y, numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
int drawSmooth, index = ccgDM_getFaceMapIndex(ccgdm, ss, f);
|
|
int origIndex;
|
|
|
|
if(ccgdm->fromEditmesh) {
|
|
origIndex = index;
|
|
} else {
|
|
origIndex = (int)ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
}
|
|
if(faceFlags) drawSmooth = (*faceFlags & ME_SMOOTH);
|
|
else drawSmooth = 1;
|
|
|
|
if (index!=-1 && (!setDrawOptions || setDrawOptions(userData, index, &drawSmooth))) {
|
|
for (S=0; S<numVerts; S++) {
|
|
VertData *faceGridData = ccgSubSurf_getFaceGridDataArray(ss, f, S);
|
|
if (drawSmooth) {
|
|
glShadeModel(GL_SMOOTH);
|
|
for (y=0; y<gridSize-1; y++) {
|
|
glBegin(GL_QUAD_STRIP);
|
|
for (x=0; x<gridSize; x++) {
|
|
VertData *a = &faceGridData[(y+0)*gridSize + x];
|
|
VertData *b = &faceGridData[(y+1)*gridSize + x];
|
|
|
|
glNormal3fv(a->no);
|
|
glVertex3fv(a->co);
|
|
glNormal3fv(b->no);
|
|
glVertex3fv(b->co);
|
|
}
|
|
glEnd();
|
|
}
|
|
} else {
|
|
glShadeModel(GL_FLAT);
|
|
glBegin(GL_QUADS);
|
|
for (y=0; y<gridSize-1; y++) {
|
|
for (x=0; x<gridSize-1; x++) {
|
|
float *a = faceGridData[(y+0)*gridSize + x].co;
|
|
float *b = faceGridData[(y+0)*gridSize + x + 1].co;
|
|
float *c = faceGridData[(y+1)*gridSize + x + 1].co;
|
|
float *d = faceGridData[(y+1)*gridSize + x].co;
|
|
float a_cX = c[0]-a[0], a_cY = c[1]-a[1], a_cZ = c[2]-a[2];
|
|
float b_dX = d[0]-b[0], b_dY = d[1]-b[1], b_dZ = d[2]-b[2];
|
|
float no[3];
|
|
|
|
no[0] = b_dY*a_cZ - b_dZ*a_cY;
|
|
no[1] = b_dZ*a_cX - b_dX*a_cZ;
|
|
no[2] = b_dX*a_cY - b_dY*a_cX;
|
|
glNormal3fv(no);
|
|
|
|
glVertex3fv(d);
|
|
glVertex3fv(c);
|
|
glVertex3fv(b);
|
|
glVertex3fv(a);
|
|
}
|
|
}
|
|
glEnd();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
static void ccgDM_drawMappedEdges(DerivedMesh *dm, int (*setDrawOptions)(void *userData, int index), void *userData) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGEdgeIterator *ei = ccgSubSurf_getEdgeIterator(ss);
|
|
int i, useAging, edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
|
|
ccgSubSurf_getUseAgeCounts(ss, &useAging, NULL, NULL, NULL);
|
|
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
VertData *edgeData = ccgSubSurf_getEdgeDataArray(ss, e);
|
|
int index = ccgDM_getEdgeMapIndex(ccgdm, ss, e);
|
|
|
|
glBegin(GL_LINE_STRIP);
|
|
if (index!=-1 && (!setDrawOptions || setDrawOptions(userData, index))) {
|
|
if (useAging && !(G.f&G_BACKBUFSEL)) {
|
|
int ageCol = 255-ccgSubSurf_getEdgeAge(ss, e)*4;
|
|
glColor3ub(0, ageCol>0?ageCol:0, 0);
|
|
}
|
|
|
|
for (i=0; i<edgeSize-1; i++) {
|
|
glVertex3fv(edgeData[i].co);
|
|
glVertex3fv(edgeData[i+1].co);
|
|
}
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
ccgEdgeIterator_free(ei);
|
|
}
|
|
static void ccgDM_drawMappedEdgesInterp(DerivedMesh *dm, int (*setDrawOptions)(void *userData, int index), void (*setDrawInterpOptions)(void *userData, int index, float t), void *userData) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGEdgeIterator *ei = ccgSubSurf_getEdgeIterator(ss);
|
|
int i, useAging, edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
|
|
ccgSubSurf_getUseAgeCounts(ss, &useAging, NULL, NULL, NULL);
|
|
|
|
for (; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
VertData *edgeData = ccgSubSurf_getEdgeDataArray(ss, e);
|
|
int index = ccgDM_getEdgeMapIndex(ccgdm, ss, e);
|
|
|
|
glBegin(GL_LINE_STRIP);
|
|
if (index!=-1 && (!setDrawOptions || setDrawOptions(userData, index))) {
|
|
for (i=0; i<edgeSize; i++) {
|
|
setDrawInterpOptions(userData, index, (float) i/(edgeSize-1));
|
|
|
|
if (useAging && !(G.f&G_BACKBUFSEL)) {
|
|
int ageCol = 255-ccgSubSurf_getEdgeAge(ss, e)*4;
|
|
glColor3ub(0, ageCol>0?ageCol:0, 0);
|
|
}
|
|
|
|
glVertex3fv(edgeData[i].co);
|
|
}
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
ccgEdgeIterator_free(ei);
|
|
}
|
|
static void ccgDM_foreachMappedFaceCenter(DerivedMesh *dm, void (*func)(void *userData, int index, float *co, float *no), void *userData) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
CCGSubSurf *ss = ccgdm->ss;
|
|
CCGFaceIterator *fi = ccgSubSurf_getFaceIterator(ss);
|
|
|
|
for (; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
int index = ccgDM_getFaceMapIndex(ccgdm, ss, f);
|
|
|
|
if (index!=-1) {
|
|
/* Face center data normal isn't updated atm. */
|
|
VertData *vd = ccgSubSurf_getFaceGridData(ss, f, 0, 0, 0);
|
|
|
|
func(userData, index, vd->co, vd->no);
|
|
}
|
|
}
|
|
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
|
|
static void ccgDM_release(DerivedMesh *dm) {
|
|
CCGDerivedMesh *ccgdm = (CCGDerivedMesh*) dm;
|
|
|
|
DM_release(dm);
|
|
|
|
if (ccgdm->dlm) displistmesh_free(ccgdm->dlm);
|
|
|
|
MEM_freeN(ccgdm->vertMap);
|
|
MEM_freeN(ccgdm->edgeMap);
|
|
MEM_freeN(ccgdm->faceMap);
|
|
MEM_freeN(ccgdm);
|
|
}
|
|
|
|
static CCGDerivedMesh *getCCGDerivedMesh(CCGSubSurf *ss, int fromEditmesh,
|
|
int drawInteriorEdges,
|
|
int useSubsurfUv, Mesh *me,
|
|
DispListMesh *dlm, DerivedMesh *dm)
|
|
{
|
|
CCGDerivedMesh *ccgdm = MEM_callocN(sizeof(*ccgdm), "ccgdm");
|
|
CCGVertIterator *vi;
|
|
CCGEdgeIterator *ei;
|
|
CCGFaceIterator *fi;
|
|
int index, totvert, totedge, totface;
|
|
int i;
|
|
int vertNum, edgeNum, faceNum;
|
|
int *vertOrigIndex, *edgeOrigIndex, *faceOrigIndex;
|
|
int *faceFlags;
|
|
int edgeSize;
|
|
int gridSize;
|
|
int gridFaces;
|
|
int gridSideVerts;
|
|
int gridInternalVerts;
|
|
int gridSideEdges;
|
|
int gridInternalEdges;
|
|
MVert *mvert = NULL;
|
|
MEdge *medge = NULL;
|
|
MFace *mface = NULL;
|
|
TFace *tface;
|
|
CCGSubSurf *uvss = NULL;
|
|
CCGFace **faceMap2Uv = NULL;
|
|
FaceVertWeight *qweight, *tweight;
|
|
|
|
if(dm) {
|
|
DM_from_template(&ccgdm->dm, dm, ccgSubSurf_getNumFinalVerts(ss),
|
|
ccgSubSurf_getNumFinalEdges(ss),
|
|
ccgSubSurf_getNumFinalFaces(ss));
|
|
tface = dm->getFaceDataArray(dm, LAYERTYPE_TFACE);
|
|
DM_add_face_layer(&ccgdm->dm, LAYERTYPE_FLAGS, LAYERFLAG_NOCOPY, NULL);
|
|
} else {
|
|
DM_init(&ccgdm->dm, ccgSubSurf_getNumFinalVerts(ss),
|
|
ccgSubSurf_getNumFinalEdges(ss),
|
|
ccgSubSurf_getNumFinalFaces(ss));
|
|
|
|
if(dlm) tface = dlm->tface;
|
|
else if(me) tface = me->tface;
|
|
else tface = NULL;
|
|
}
|
|
|
|
ccgdm->dm.getMinMax = ccgDM_getMinMax;
|
|
ccgdm->dm.getNumVerts = ccgDM_getNumVerts;
|
|
ccgdm->dm.getNumFaces = ccgDM_getNumFaces;
|
|
|
|
ccgdm->dm.getNumEdges = ccgDM_getNumEdges;
|
|
ccgdm->dm.getVert = ccgDM_getFinalVert;
|
|
ccgdm->dm.getEdge = ccgDM_getFinalEdge;
|
|
ccgdm->dm.getFace = ccgDM_getFinalFace;
|
|
ccgdm->dm.getVertArray = ccgDM_getFinalVertArray;
|
|
ccgdm->dm.getEdgeArray = ccgDM_getFinalEdgeArray;
|
|
ccgdm->dm.getFaceArray = ccgDM_getFinalFaceArray;
|
|
ccgdm->dm.getVertData = DM_get_vert_data;
|
|
ccgdm->dm.getEdgeData = DM_get_edge_data;
|
|
ccgdm->dm.getFaceData = DM_get_face_data;
|
|
ccgdm->dm.getVertDataArray = DM_get_vert_data_layer;
|
|
ccgdm->dm.getEdgeDataArray = DM_get_edge_data_layer;
|
|
ccgdm->dm.getFaceDataArray = DM_get_face_data_layer;
|
|
|
|
ccgdm->dm.getVertCos = ccgdm_getVertCos;
|
|
ccgdm->dm.foreachMappedVert = ccgDM_foreachMappedVert;
|
|
ccgdm->dm.foreachMappedEdge = ccgDM_foreachMappedEdge;
|
|
ccgdm->dm.foreachMappedFaceCenter = ccgDM_foreachMappedFaceCenter;
|
|
ccgdm->dm.convertToDispListMesh = ccgDM_convertToDispListMesh;
|
|
|
|
ccgdm->dm.drawVerts = ccgDM_drawVerts;
|
|
ccgdm->dm.drawEdges = ccgDM_drawEdges;
|
|
ccgdm->dm.drawLooseEdges = ccgDM_drawLooseEdges;
|
|
ccgdm->dm.drawFacesSolid = ccgDM_drawFacesSolid;
|
|
ccgdm->dm.drawFacesColored = ccgDM_drawFacesColored;
|
|
ccgdm->dm.drawFacesTex = ccgDM_drawFacesTex;
|
|
ccgdm->dm.drawMappedFaces = ccgDM_drawMappedFaces;
|
|
ccgdm->dm.drawMappedFacesTex = ccgDM_drawMappedFacesTex;
|
|
|
|
ccgdm->dm.drawMappedEdgesInterp = ccgDM_drawMappedEdgesInterp;
|
|
ccgdm->dm.drawMappedEdges = ccgDM_drawMappedEdges;
|
|
|
|
ccgdm->dm.release = ccgDM_release;
|
|
|
|
ccgdm->ss = ss;
|
|
ccgdm->fromEditmesh = fromEditmesh;
|
|
ccgdm->drawInteriorEdges = drawInteriorEdges;
|
|
ccgdm->useSubsurfUv = useSubsurfUv;
|
|
ccgdm->me = me;
|
|
ccgdm->dlm = dlm;
|
|
|
|
totvert = ccgSubSurf_getNumVerts(ss);
|
|
ccgdm->vertMap = MEM_mallocN(totvert * sizeof(*ccgdm->vertMap), "vertMap");
|
|
vi = ccgSubSurf_getVertIterator(ss);
|
|
for(; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
|
|
if(fromEditmesh) {
|
|
ccgdm->vertMap[ccgDM_getVertMapIndex(ccgdm, ss, v)].vert = v;
|
|
} else {
|
|
ccgdm->vertMap[(int) ccgSubSurf_getVertVertHandle(ss, v)].vert = v;
|
|
}
|
|
}
|
|
ccgVertIterator_free(vi);
|
|
|
|
totedge = ccgSubSurf_getNumEdges(ss);
|
|
ccgdm->edgeMap = MEM_mallocN(totedge * sizeof(*ccgdm->edgeMap), "edgeMap");
|
|
ei = ccgSubSurf_getEdgeIterator(ss);
|
|
for(; !ccgEdgeIterator_isStopped(ei); ccgEdgeIterator_next(ei)) {
|
|
CCGEdge *e = ccgEdgeIterator_getCurrent(ei);
|
|
|
|
if(fromEditmesh) {
|
|
ccgdm->edgeMap[ccgDM_getEdgeMapIndex(ccgdm,ss,e)].edge = e;
|
|
} else {
|
|
ccgdm->edgeMap[(int) ccgSubSurf_getEdgeEdgeHandle(ss, e)].edge = e;
|
|
}
|
|
}
|
|
|
|
totface = ccgSubSurf_getNumFaces(ss);
|
|
ccgdm->faceMap = MEM_mallocN(totface * sizeof(*ccgdm->faceMap), "faceMap");
|
|
fi = ccgSubSurf_getFaceIterator(ss);
|
|
for(; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
|
|
if(fromEditmesh) {
|
|
ccgdm->faceMap[ccgDM_getFaceMapIndex(ccgdm,ss,f)].face = f;
|
|
} else {
|
|
ccgdm->faceMap[(int) ccgSubSurf_getFaceFaceHandle(ss, f)].face = f;
|
|
}
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
|
|
if(useSubsurfUv && tface) {
|
|
/* slightly dodgy hack to use current ss_sync_from_uv function */
|
|
DispListMesh dlm;
|
|
|
|
dlm.mface = dm->dupFaceArray(dm);
|
|
dlm.tface = tface;
|
|
dlm.mvert = dm->dupVertArray(dm);
|
|
dlm.totvert = dm->getNumVerts(dm);
|
|
dlm.totface = dm->getNumFaces(dm);
|
|
|
|
/* not for editmesh currently */
|
|
uvss = _getSubSurf(NULL, ccgSubSurf_getSubdivisionLevels(ss),
|
|
0, 1, 0);
|
|
|
|
if(ss_sync_from_uv(uvss, ss, NULL, &dlm)) {
|
|
faceMap2Uv = MEM_mallocN(totface * sizeof(*faceMap2Uv),
|
|
"facemapuv");
|
|
|
|
fi = ccgSubSurf_getFaceIterator(uvss);
|
|
for(; !ccgFaceIterator_isStopped(fi); ccgFaceIterator_next(fi)) {
|
|
CCGFace *f = ccgFaceIterator_getCurrent(fi);
|
|
faceMap2Uv[(int) ccgSubSurf_getFaceFaceHandle(uvss, f)] = f;
|
|
}
|
|
ccgFaceIterator_free(fi);
|
|
}
|
|
|
|
MEM_freeN(dlm.mface);
|
|
MEM_freeN(dlm.mvert);
|
|
}
|
|
|
|
edgeSize = ccgSubSurf_getEdgeSize(ss);
|
|
gridSize = ccgSubSurf_getGridSize(ss);
|
|
gridFaces = gridSize - 1;
|
|
gridSideVerts = gridSize - 2;
|
|
gridInternalVerts = gridSideVerts * gridSideVerts;
|
|
gridSideEdges = gridSize - 1;
|
|
gridInternalEdges = (gridSideEdges - 1) * gridSideEdges * 2;
|
|
|
|
calc_ss_weights(gridFaces, &qweight, &tweight);
|
|
|
|
vertNum = 0;
|
|
edgeNum = 0;
|
|
faceNum = 0;
|
|
|
|
if(dm) {
|
|
mvert = dm->dupVertArray(dm);
|
|
medge = dm->dupEdgeArray(dm);
|
|
mface = dm->dupFaceArray(dm);
|
|
} else if(dlm) {
|
|
mvert = dlm->mvert;
|
|
medge = dlm->medge;
|
|
mface = dlm->mface;
|
|
} else if(me) {
|
|
mvert = me->mvert;
|
|
medge = me->medge;
|
|
mface = me->mface;
|
|
}
|
|
|
|
vertOrigIndex = DM_get_vert_data_layer(&ccgdm->dm, LAYERTYPE_ORIGINDEX);
|
|
edgeOrigIndex = DM_get_edge_data_layer(&ccgdm->dm, LAYERTYPE_ORIGINDEX);
|
|
faceOrigIndex = DM_get_face_data_layer(&ccgdm->dm, LAYERTYPE_ORIGINDEX);
|
|
|
|
faceFlags = DM_get_face_data_layer(&ccgdm->dm, LAYERTYPE_FLAGS);
|
|
|
|
for(index = 0; index < totface; ++index) {
|
|
CCGFace *f = ccgdm->faceMap[index].face;
|
|
CCGFace *uvf = faceMap2Uv ? faceMap2Uv[index] : NULL;
|
|
int numVerts = ccgSubSurf_getFaceNumVerts(ss, f);
|
|
int numFinalEdges = numVerts * (gridSideEdges + gridInternalEdges);
|
|
int mapIndex = ccgDM_getFaceMapIndex(ccgdm, ss, f);
|
|
int origIndex = (int)ccgSubSurf_getFaceFaceHandle(ss, f);
|
|
FaceVertWeight *weight = (numVerts == 4) ? qweight : tweight;
|
|
int S, x, y;
|
|
int vertIdx[4];
|
|
|
|
ccgdm->faceMap[index].startVert = vertNum;
|
|
ccgdm->faceMap[index].startEdge = edgeNum;
|
|
ccgdm->faceMap[index].startFace = faceNum;
|
|
|
|
/* set the face base vert */
|
|
*((int*)ccgSubSurf_getFaceUserData(ss, f)) = vertNum;
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
CCGVert *v = ccgSubSurf_getFaceVert(ss, f, S);
|
|
|
|
if(fromEditmesh)
|
|
vertIdx[S] = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
else
|
|
vertIdx[S] = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
}
|
|
|
|
DM_interp_vert_data(dm, &ccgdm->dm, vertIdx, weight[0][0],
|
|
numVerts, vertNum);
|
|
*vertOrigIndex = ORIGINDEX_NONE;
|
|
++vertOrigIndex;
|
|
++vertNum;
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
int prevS = (S - 1 + numVerts) % numVerts;
|
|
int nextS = (S + 1) % numVerts;
|
|
int otherS = (numVerts == 4) ? (S + 2) % numVerts : 3;
|
|
for(x = 1; x < gridFaces; x++) {
|
|
float w[4];
|
|
w[prevS] = weight[x][0][0];
|
|
w[S] = weight[x][0][1];
|
|
w[nextS] = weight[x][0][2];
|
|
w[otherS] = weight[x][0][3];
|
|
DM_interp_vert_data(dm, &ccgdm->dm, vertIdx, w,
|
|
numVerts, vertNum);
|
|
*vertOrigIndex = ORIGINDEX_NONE;
|
|
++vertOrigIndex;
|
|
++vertNum;
|
|
}
|
|
}
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
int prevS = (S - 1 + numVerts) % numVerts;
|
|
int nextS = (S + 1) % numVerts;
|
|
int otherS = (numVerts == 4) ? (S + 2) % numVerts : 3;
|
|
for(y = 1; y < gridFaces; y++) {
|
|
for(x = 1; x < gridFaces; x++) {
|
|
float w[4];
|
|
w[prevS] = weight[y * gridFaces + x][0][0];
|
|
w[S] = weight[y * gridFaces + x][0][1];
|
|
w[nextS] = weight[y * gridFaces + x][0][2];
|
|
w[otherS] = weight[y * gridFaces + x][0][3];
|
|
DM_interp_vert_data(dm, &ccgdm->dm, vertIdx, w,
|
|
numVerts, vertNum);
|
|
*vertOrigIndex = ORIGINDEX_NONE;
|
|
++vertOrigIndex;
|
|
++vertNum;
|
|
}
|
|
}
|
|
}
|
|
|
|
for(i = 0; i < numFinalEdges; ++i)
|
|
*(int *)DM_get_edge_data(&ccgdm->dm, edgeNum + i,
|
|
LAYERTYPE_ORIGINDEX) = ORIGINDEX_NONE;
|
|
|
|
for(S = 0; S < numVerts; S++) {
|
|
int prevS = (S - 1 + numVerts) % numVerts;
|
|
int nextS = (S + 1) % numVerts;
|
|
int otherS = (numVerts == 4) ? (S + 2) % numVerts : 3;
|
|
|
|
weight = (numVerts == 4) ? qweight : tweight;
|
|
|
|
for(y = 0; y < gridFaces; y++) {
|
|
for(x = 0; x < gridFaces; x++) {
|
|
FaceVertWeight w;
|
|
int j;
|
|
|
|
for(j = 0; j < 4; ++j) {
|
|
w[j][prevS] = (*weight)[j][0];
|
|
w[j][S] = (*weight)[j][1];
|
|
w[j][nextS] = (*weight)[j][2];
|
|
w[j][otherS] = (*weight)[j][3];
|
|
}
|
|
|
|
DM_interp_face_data(dm, &ccgdm->dm, &origIndex, NULL,
|
|
&w, 1, faceNum);
|
|
weight++;
|
|
|
|
if(uvf) {
|
|
TFace *tf = DM_get_face_data(&ccgdm->dm, faceNum,
|
|
LAYERTYPE_TFACE);
|
|
float *newuv;
|
|
|
|
newuv = getFaceUV(uvss, uvf, S, x + 0, y + 0,
|
|
edgeSize, gridSize);
|
|
tf->uv[0][0] = newuv[0]; tf->uv[0][1] = newuv[1];
|
|
newuv = getFaceUV(uvss, uvf, S, x + 0, y + 1,
|
|
edgeSize, gridSize);
|
|
tf->uv[1][0] = newuv[0]; tf->uv[1][1] = newuv[1];
|
|
newuv = getFaceUV(uvss, uvf, S, x + 1, y + 1,
|
|
edgeSize, gridSize);
|
|
tf->uv[2][0] = newuv[0]; tf->uv[2][1] = newuv[1];
|
|
newuv = getFaceUV(uvss, uvf, S, x + 1, y + 0,
|
|
edgeSize, gridSize);
|
|
tf->uv[3][0] = newuv[0]; tf->uv[3][1] = newuv[1];
|
|
}
|
|
|
|
*faceOrigIndex = mapIndex;
|
|
if(mface)
|
|
*faceFlags = mface[origIndex].flag;
|
|
else
|
|
*faceFlags = ME_SMOOTH;
|
|
|
|
++faceOrigIndex;
|
|
++faceFlags;
|
|
++faceNum;
|
|
}
|
|
}
|
|
}
|
|
|
|
edgeNum += numFinalEdges;
|
|
}
|
|
|
|
if(uvss) {
|
|
ccgSubSurf_free(uvss);
|
|
MEM_freeN(faceMap2Uv);
|
|
}
|
|
|
|
for(index = 0; index < totedge; ++index) {
|
|
CCGEdge *e = ccgdm->edgeMap[index].edge;
|
|
int numFinalEdges = edgeSize - 1;
|
|
int mapIndex = ccgDM_getEdgeMapIndex(ccgdm, ss, e);
|
|
int x;
|
|
int vertIdx[2];
|
|
|
|
if(fromEditmesh) {
|
|
CCGVert *v;
|
|
v = ccgSubSurf_getEdgeVert0(ss, e);
|
|
vertIdx[0] = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
v = ccgSubSurf_getEdgeVert1(ss, e);
|
|
vertIdx[1] = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
} else {
|
|
CCGVert *v;
|
|
v = ccgSubSurf_getEdgeVert0(ss, e);
|
|
vertIdx[0] = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
v = ccgSubSurf_getEdgeVert1(ss, e);
|
|
vertIdx[1] = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
}
|
|
|
|
ccgdm->edgeMap[index].startVert = vertNum;
|
|
ccgdm->edgeMap[index].startEdge = edgeNum;
|
|
|
|
/* set the edge base vert */
|
|
*((int*)ccgSubSurf_getEdgeUserData(ss, e)) = vertNum;
|
|
|
|
for(x = 1; x < edgeSize - 1; x++) {
|
|
float w[2];
|
|
w[1] = (float) x / (edgeSize - 1);
|
|
w[0] = 1 - w[1];
|
|
DM_interp_vert_data(dm, &ccgdm->dm, vertIdx, w, 2, vertNum);
|
|
*vertOrigIndex = ORIGINDEX_NONE;
|
|
++vertOrigIndex;
|
|
++vertNum;
|
|
}
|
|
|
|
for(i = 0; i < numFinalEdges; ++i)
|
|
*(int *)DM_get_edge_data(&ccgdm->dm, edgeNum + i,
|
|
LAYERTYPE_ORIGINDEX) = mapIndex;
|
|
|
|
edgeNum += numFinalEdges;
|
|
}
|
|
|
|
for(index = 0; index < totvert; ++index) {
|
|
CCGVert *v = ccgdm->vertMap[index].vert;
|
|
int mapIndex = ccgDM_getVertMapIndex(ccgdm, ccgdm->ss, v);
|
|
int vertIdx;
|
|
|
|
if(fromEditmesh)
|
|
vertIdx = ccgDM_getVertMapIndex(NULL, ss, v);
|
|
else
|
|
vertIdx = (int)ccgSubSurf_getVertVertHandle(ss, v);
|
|
|
|
ccgdm->vertMap[index].startVert = vertNum;
|
|
|
|
/* set the vert base vert */
|
|
*((int*) ccgSubSurf_getVertUserData(ss, v)) = vertNum;
|
|
|
|
DM_copy_vert_data(dm, &ccgdm->dm, vertIdx, vertNum, 1);
|
|
*vertOrigIndex = mapIndex;
|
|
++vertOrigIndex;
|
|
++vertNum;
|
|
}
|
|
|
|
if(dm) {
|
|
MEM_freeN(mvert);
|
|
MEM_freeN(medge);
|
|
MEM_freeN(mface);
|
|
}
|
|
|
|
MEM_freeN(qweight);
|
|
MEM_freeN(tweight);
|
|
|
|
return ccgdm;
|
|
}
|
|
|
|
/***/
|
|
|
|
DerivedMesh *subsurf_make_derived_from_editmesh(EditMesh *em, SubsurfModifierData *smd, float (*vertCos)[3]) {
|
|
int useSimple = smd->subdivType==ME_SIMPLE_SUBSURF;
|
|
int useAging = smd->flags&eSubsurfModifierFlag_DebugIncr;
|
|
int drawInteriorEdges = !(smd->flags&eSubsurfModifierFlag_ControlEdges);
|
|
|
|
smd->emCache = _getSubSurf(smd->emCache, smd->levels, useAging, 0, useSimple);
|
|
ss_sync_from_editmesh(smd->emCache, em, vertCos, useSimple);
|
|
|
|
return (DerivedMesh*) getCCGDerivedMesh(smd->emCache, 1, drawInteriorEdges, 0, NULL, NULL, NULL);
|
|
}
|
|
|
|
DerivedMesh *subsurf_make_derived_from_dlm_em(DispListMesh *dlm, SubsurfModifierData *smd, float (*vertCos)[3]) {
|
|
int useSimple = smd->subdivType==ME_SIMPLE_SUBSURF;
|
|
int useAging = smd->flags&eSubsurfModifierFlag_DebugIncr;
|
|
int useSubsurfUv = smd->flags&eSubsurfModifierFlag_SubsurfUv;
|
|
int drawInteriorEdges = !(smd->flags&eSubsurfModifierFlag_ControlEdges);
|
|
|
|
smd->emCache = _getSubSurf(smd->emCache, smd->levels, useAging, 0, useSimple);
|
|
|
|
ss_sync_from_mesh(smd->emCache, NULL, dlm, vertCos, useSimple);
|
|
|
|
return (DerivedMesh*) getCCGDerivedMesh(smd->emCache, 0, drawInteriorEdges, useSubsurfUv, NULL, dlm, NULL);
|
|
}
|
|
|
|
DerivedMesh *subsurf_make_derived_from_mesh(Mesh *me, DispListMesh *dlm, SubsurfModifierData *smd, int useRenderParams, float (*vertCos)[3], int isFinalCalc) {
|
|
int useSimple = smd->subdivType==ME_SIMPLE_SUBSURF;
|
|
int useSubsurfUv = smd->flags&eSubsurfModifierFlag_SubsurfUv;
|
|
int drawInteriorEdges = !(smd->flags&eSubsurfModifierFlag_ControlEdges);
|
|
DispListMesh *ndlm;
|
|
|
|
/* Do not use cache in render mode. */
|
|
if (useRenderParams) {
|
|
CCGSubSurf *ss = _getSubSurf(NULL, smd->renderLevels, 0, 1, useSimple);
|
|
|
|
ss_sync_from_mesh(ss, me, dlm, vertCos, useSimple);
|
|
|
|
ndlm = ss_to_displistmesh(ss, NULL, 0, drawInteriorEdges, useSubsurfUv, me, dlm);
|
|
if (dlm) displistmesh_free(dlm);
|
|
|
|
ccgSubSurf_free(ss);
|
|
|
|
return derivedmesh_from_displistmesh(ndlm, NULL);
|
|
} else {
|
|
int useIncremental = (smd->flags&eSubsurfModifierFlag_Incremental);
|
|
int useAging = smd->flags&eSubsurfModifierFlag_DebugIncr;
|
|
CCGSubSurf *ss;
|
|
|
|
/* It is quite possible there is a much better place to do this. It
|
|
* depends a bit on how rigourously we expect this function to never
|
|
* be called in editmode. In semi-theory we could share a single
|
|
* cache, but the handles used inside and outside editmode are not
|
|
* the same so we would need some way of converting them. Its probably
|
|
* not worth the effort. But then why am I even writing this long
|
|
* comment that no one will read? Hmmm. - zr
|
|
*/
|
|
if (smd->emCache) {
|
|
ccgSubSurf_free(smd->emCache);
|
|
smd->emCache = NULL;
|
|
}
|
|
|
|
if (useIncremental && isFinalCalc) {
|
|
smd->mCache = ss = _getSubSurf(smd->mCache, smd->levels, useAging, 0, useSimple);
|
|
|
|
ss_sync_from_mesh(ss, me, dlm, vertCos, useSimple);
|
|
|
|
return (DerivedMesh*) getCCGDerivedMesh(ss, 0, drawInteriorEdges, useSubsurfUv, me, dlm, NULL);
|
|
} else {
|
|
if (smd->mCache && isFinalCalc) {
|
|
ccgSubSurf_free(smd->mCache);
|
|
smd->mCache = NULL;
|
|
}
|
|
|
|
ss = _getSubSurf(NULL, smd->levels, 0, 1, useSimple);
|
|
ss_sync_from_mesh(ss, me, dlm, vertCos, useSimple);
|
|
|
|
ndlm = ss_to_displistmesh(ss, NULL, 0, drawInteriorEdges, useSubsurfUv, me, dlm);
|
|
|
|
if (dlm) displistmesh_free(dlm);
|
|
ccgSubSurf_free(ss);
|
|
|
|
return derivedmesh_from_displistmesh(ndlm, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct DerivedMesh *subsurf_make_derived_from_derived(
|
|
struct DerivedMesh *dm,
|
|
struct SubsurfModifierData *smd,
|
|
int useRenderParams, float (*vertCos)[3],
|
|
int isFinalCalc, int editMode)
|
|
{
|
|
int useSimple = smd->subdivType == ME_SIMPLE_SUBSURF;
|
|
int useAging = smd->flags & eSubsurfModifierFlag_DebugIncr;
|
|
int useSubsurfUv = smd->flags & eSubsurfModifierFlag_SubsurfUv;
|
|
int drawInteriorEdges = !(smd->flags & eSubsurfModifierFlag_ControlEdges);
|
|
DerivedMesh *result;
|
|
|
|
if(editMode) {
|
|
smd->emCache = _getSubSurf(smd->emCache, smd->levels, useAging, 0,
|
|
useSimple);
|
|
ss_sync_from_derivedmesh(smd->emCache, dm, vertCos, useSimple);
|
|
|
|
return (DerivedMesh *)getCCGDerivedMesh(smd->emCache, 0,
|
|
drawInteriorEdges,
|
|
useSubsurfUv, NULL, NULL, dm);
|
|
} else if(useRenderParams) {
|
|
/* Do not use cache in render mode. */
|
|
CCGSubSurf *ss = _getSubSurf(NULL, smd->renderLevels, 0, 1, useSimple);
|
|
|
|
ss_sync_from_derivedmesh(ss, dm, vertCos, useSimple);
|
|
|
|
result = ss_to_cdderivedmesh(ss, 0, drawInteriorEdges,
|
|
useSubsurfUv, dm);
|
|
|
|
ccgSubSurf_free(ss);
|
|
|
|
return result;
|
|
} else {
|
|
int useIncremental = (smd->flags & eSubsurfModifierFlag_Incremental);
|
|
int useAging = smd->flags & eSubsurfModifierFlag_DebugIncr;
|
|
CCGSubSurf *ss;
|
|
|
|
/* It is quite possible there is a much better place to do this. It
|
|
* depends a bit on how rigourously we expect this function to never
|
|
* be called in editmode. In semi-theory we could share a single
|
|
* cache, but the handles used inside and outside editmode are not
|
|
* the same so we would need some way of converting them. Its probably
|
|
* not worth the effort. But then why am I even writing this long
|
|
* comment that no one will read? Hmmm. - zr
|
|
*/
|
|
if(smd->emCache) {
|
|
ccgSubSurf_free(smd->emCache);
|
|
smd->emCache = NULL;
|
|
}
|
|
|
|
if(useIncremental && isFinalCalc) {
|
|
smd->mCache = ss = _getSubSurf(smd->mCache, smd->levels,
|
|
useAging, 0, useSimple);
|
|
|
|
ss_sync_from_derivedmesh(ss, dm, vertCos, useSimple);
|
|
|
|
return ss_to_cdderivedmesh(ss, 0, drawInteriorEdges,
|
|
useSubsurfUv, dm);
|
|
} else {
|
|
if (smd->mCache && isFinalCalc) {
|
|
ccgSubSurf_free(smd->mCache);
|
|
smd->mCache = NULL;
|
|
}
|
|
|
|
ss = _getSubSurf(NULL, smd->levels, 0, 1, useSimple);
|
|
ss_sync_from_derivedmesh(ss, dm, vertCos, useSimple);
|
|
|
|
result = ss_to_cdderivedmesh(ss, 0, drawInteriorEdges,
|
|
useSubsurfUv, dm);
|
|
|
|
ccgSubSurf_free(ss);
|
|
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
|
|
void subsurf_calculate_limit_positions(Mesh *me, float (*positions_r)[3])
|
|
{
|
|
/* Finds the subsurf limit positions for the verts in a mesh
|
|
* and puts them in an array of floats. Please note that the
|
|
* calculated vert positions is incorrect for the verts
|
|
* on the boundary of the mesh.
|
|
*/
|
|
CCGSubSurf *ss = _getSubSurf(NULL, 1, 0, 1, 0);
|
|
float edge_sum[3], face_sum[3];
|
|
CCGVertIterator *vi;
|
|
|
|
ss_sync_from_mesh(ss, me, NULL, NULL, 0);
|
|
|
|
vi = ccgSubSurf_getVertIterator(ss);
|
|
for (; !ccgVertIterator_isStopped(vi); ccgVertIterator_next(vi)) {
|
|
CCGVert *v = ccgVertIterator_getCurrent(vi);
|
|
int idx = (int) ccgSubSurf_getVertVertHandle(ss, v);
|
|
int N = ccgSubSurf_getVertNumEdges(ss, v);
|
|
int numFaces = ccgSubSurf_getVertNumFaces(ss, v);
|
|
float *co;
|
|
int i;
|
|
|
|
edge_sum[0]= edge_sum[1]= edge_sum[2]= 0.0;
|
|
face_sum[0]= face_sum[1]= face_sum[2]= 0.0;
|
|
|
|
for (i=0; i<N; i++) {
|
|
CCGEdge *e = ccgSubSurf_getVertEdge(ss, v, i);
|
|
VecAddf(edge_sum, edge_sum, ccgSubSurf_getEdgeData(ss, e, 1));
|
|
}
|
|
for (i=0; i<numFaces; i++) {
|
|
CCGFace *f = ccgSubSurf_getVertFace(ss, v, i);
|
|
VecAddf(face_sum, face_sum, ccgSubSurf_getFaceCenterData(ss, f));
|
|
}
|
|
|
|
co = ccgSubSurf_getVertData(ss, v);
|
|
positions_r[idx][0] = (co[0]*N*N + edge_sum[0]*4 + face_sum[0])/(N*(N+5));
|
|
positions_r[idx][1] = (co[1]*N*N + edge_sum[1]*4 + face_sum[1])/(N*(N+5));
|
|
positions_r[idx][2] = (co[2]*N*N + edge_sum[2]*4 + face_sum[2])/(N*(N+5));
|
|
}
|
|
ccgVertIterator_free(vi);
|
|
|
|
ccgSubSurf_free(ss);
|
|
}
|
|
|