*/}}

la_tns_mesh.c 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829
  1. /*
  2. * LaGUI: A graphical application framework.
  3. * Copyright (C) 2022-2023 Wu Yiming
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. #include "la_5.h"
  19. #include <math.h>
  20. extern tnsMain *T;
  21. void tnsMMeshClearFirstLastSelection(tnsMeshObject* mo);
  22. void tnsPrintMeshTopology(tnsMeshObject*mo){
  23. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ printf("%d ",mv->i); } printf("\n");
  24. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ printf("%d-%d ",me->vl->i,me->vr->i); } printf("\n");
  25. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ printf("%d-[",mf->looplen);
  26. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){ tnsMEdge*ie=lip->p;
  27. printf("%d%s%d ",ie->vl->i,(ie->fl&&ie->fr)?":":(ie->fl?"l":(ie->fr?"r":"-")),ie->vr->i); }
  28. printf("] ");
  29. }printf("\n");
  30. }
  31. tnsEdgeHash* tnsCreateEdgeHash(int OptionalInitialVertCount){
  32. tnsEdgeHash*eh=memAcquireSimple(sizeof(tnsEdgeHash));
  33. eh->max=OptionalInitialVertCount;
  34. if(eh->max<16) eh->max=16;
  35. arrInitLength(&eh->vl,eh->max,&eh->max, sizeof(tnsEdgeHashVert));
  36. return eh;
  37. }
  38. void tnsDestroyEdgeHash(tnsEdgeHash* eh){
  39. for(int i=0;i<eh->max;i++){
  40. tnsEdgeHashVert* ehv=&eh->vl[i];
  41. if(ehv->e) free(ehv->e);
  42. }
  43. if(eh->vl) free(eh->vl);
  44. memFree(eh);
  45. }
  46. void tnsEdgeHashAddVertPair(tnsEdgeHash* eh, int v1, int v2){
  47. if(v1==v2) return; if(v1>v2) LA_SWAP(int, v1,v2); if(v1>=eh->max) return;
  48. tnsEdgeHashVert* ehv=&eh->vl[v1];
  49. if(!ehv->e) arrInitLength(&ehv->e,1,&ehv->max, sizeof(tnsEdgeHashEdge));
  50. else arrEnsureLength(&ehv->e, ehv->next, &ehv->max, sizeof(tnsEdgeHashEdge));
  51. for(int i=0;i<ehv->next;i++){ if(ehv->e[i].tv==v2) return; }
  52. ehv->e[ehv->next].tv=v2;
  53. ehv->next++;
  54. }
  55. tnsEdgeHashEdge* tnsEdgeHashGetEdge(tnsEdgeHash* eh, int v1, int v2){
  56. if(v1==v2) return 0; if(v1>v2) LA_SWAP(int, v1,v2); if(v1>=eh->max) return 0;
  57. tnsEdgeHashVert* ehv=&eh->vl[v1];
  58. for(int i=0;i<ehv->next;i++){ if(ehv->e[i].tv==v2) return &ehv->e[i]; }
  59. return 0;
  60. }
  61. void tnsInitMesh(tnsMeshObject* mo, int Initialv, int Initiale, int Initialf){
  62. arrInitLength(&mo->v, Initialv, &mo->maxv, sizeof(tnsVert));
  63. arrInitLength(&mo->e, Initiale, &mo->maxe, sizeof(tnsEdge));
  64. arrInitLength(&mo->f, Initialf, &mo->maxf, sizeof(tnsFace));
  65. }
  66. tnsVert* tnsFillVertI(tnsMeshObject* mo, int index, real x, real y, real z){
  67. arrEnsureLength(&mo->v, index, &mo->maxv, sizeof(tnsVert));
  68. mo->v[index].p[0]=x;
  69. mo->v[index].p[1]=y;
  70. mo->v[index].p[2]=z;
  71. return &mo->v[index];
  72. }
  73. tnsVert* tnsFillVert(tnsMeshObject* mo, real x, real y, real z){
  74. int index=mo->totv; mo->totv++;
  75. return tnsFillVertI(mo, index, x,y,z);
  76. }
  77. tnsFace* tnsFillFace(tnsMeshObject* mo, int vertcount, ...){
  78. arrEnsureLength(&mo->f, mo->totf, &mo->maxf, sizeof(tnsFace));
  79. arrInitLength(&mo->f[mo->totf].loop, vertcount, &mo->f[mo->totf].looplen, sizeof(int));
  80. va_list list; va_start(list, vertcount); int id=va_arg(list, int); if(id==-1){ va_end(list); mo->totf++; return &mo->f[mo->totf-1]; }
  81. for(int i=0;i<vertcount;i++){ mo->f[mo->totf].loop[i]=id; id=va_arg(list, int); }
  82. va_end(list);
  83. mo->totf++; return &mo->f[mo->totf-1];
  84. }
  85. void tnsFillFaceLoop(tnsFace* f, int i, int v){ f->loop[i]=v; }
  86. void tnsFillFaceNormal(tnsFace* f, real* normal){ tnsVectorSet3v(f->n, normal); }
  87. void tnsFillVertNormal(tnsVert* v, real* normal){ tnsVectorSet3v(v->n, normal); }
  88. int tnsMergeMeshObjects(tnsMeshObject* into, tnsMeshObject* mo){
  89. if(into->Base.Type!=TNS_OBJECT_MESH||mo->Base.Type!=TNS_OBJECT_MESH) return 0;
  90. if(!mo->totv){ tnsDestroyObject(mo); return 1; }
  91. tnsVector3d v,vf; tnsMatrix44d inv; tnsInverse44d(inv, into->Base.GlobalTransform);
  92. for(int i=0;i<mo->totv;i++){
  93. tnsVectorSet3v(v,mo->v[i].p); tnsApplyTransform43d(vf, mo->Base.GlobalTransform, v);
  94. tnsApplyTransform43d(v, inv, vf); tnsVectorSet3v(mo->v[i].p,v);
  95. }
  96. for(int i=0;i<mo->tote;i++){ mo->e[i].l+=into->totv; mo->e[i].r+=into->totv; }
  97. for(int i=0;i<mo->totf;i++){ for(int l=0;l<mo->f[i].looplen;l++){ mo->f[i].loop[l]+=into->totv; } }
  98. int origv=into->totv, orige=into->tote, origf=into->totf;
  99. into->maxv=into->totv; into->maxe=into->tote; into->maxf=into->totf;
  100. into->totv+=mo->totv; into->tote+=mo->tote; into->totf+=mo->totf;
  101. arrEnsureLength(&into->v, into->totv, &into->maxv, sizeof(tnsVert)); if(mo->totv) memcpy(&into->v[origv],mo->v,sizeof(tnsVert)*mo->totv);
  102. arrEnsureLength(&into->e, into->tote, &into->maxe, sizeof(tnsEdge)); if(mo->tote) memcpy(&into->e[orige],mo->e,sizeof(tnsEdge)*mo->tote);
  103. arrEnsureLength(&into->f, into->totf, &into->maxf, sizeof(tnsFace)); if(mo->totf){ memcpy(&into->f[origf],mo->f,sizeof(tnsFace)*mo->totf);
  104. for(int i=origf;i<into->totf;i++){ into->f[i].loop=0; arrInitLength(&into->f[i].loop, into->f[i].looplen, &into->f[i].looplen, sizeof(int));
  105. for(int l=0;l<into->f[i].looplen;l++){ into->f[i].loop[l]=mo->f[i-origf].loop[l]; }
  106. }
  107. }
  108. tnsDestroyObject(mo); return 1;
  109. }
  110. tnsMeshObject* tnsDuplicateMeshObject(tnsMeshObject* from){
  111. if(from->Base.Type!=TNS_OBJECT_MESH) return 0;
  112. tnsMeshObject* to = memAcquireHyper(sizeof(tnsMeshObject));
  113. tnsInitObjectBase(to, from->Base.ParentObject?from->Base.ParentObject:from->Base.InRoot, from->Base.Name->Ptr, TNS_OBJECT_MESH,0,0,0,0,0,0,0,0,1);
  114. tnsCopyObjectTransformationsLocal(to,from);
  115. if(!from->totv){ return to; }
  116. to->totv=from->totv; to->tote=from->tote; to->totf=from->totf;
  117. arrInitLength(&to->v, to->totv, &to->maxv, sizeof(tnsVert)); if(from->totv) memcpy(to->v,from->v,sizeof(tnsVert)*from->totv);
  118. arrInitLength(&to->e, to->tote, &to->maxe, sizeof(tnsEdge)); if(from->tote) memcpy(to->e,from->e,sizeof(tnsEdge)*from->tote);
  119. arrInitLength(&to->f, to->totf, &to->maxf, sizeof(tnsFace)); if(from->totf) memcpy(to->f,from->f,sizeof(tnsFace)*from->totf);
  120. for(int i=0;i<to->totf;i++){ to->f[i].loop=0; arrInitLength(&to->f[i].loop, to->f[i].looplen, &to->f[i].looplen, sizeof(int));
  121. for(int l=0;l<to->f[i].looplen;l++){ to->f[i].loop[l]=from->f[i].loop[l]; }
  122. }
  123. return to;
  124. }
  125. void tnsInitMeshPlane(tnsMeshObject* mo, real size){
  126. tnsInitMesh(mo, 4,0,1);
  127. tnsFillVert(mo, size, size,0); tnsFillVert(mo,-size, size,0);
  128. tnsFillVert(mo,-size,-size,0); tnsFillVert(mo, size,-size,0);
  129. tnsFace* f=tnsFillFace(mo, 4, 0,1,2,3); tnsVector3d n={0,0,1}; tnsFillFaceNormal(f,n);
  130. mo->v[0].flags|=TNS_MESH_FLAG_SELECTED;
  131. mo->v[1].flags|=TNS_MESH_FLAG_SELECTED;
  132. mo->v[2].flags|=TNS_MESH_FLAG_SELECTED;
  133. mo->v[3].flags|=TNS_MESH_FLAG_SELECTED;
  134. tnsMMeshEnsureSelectionFromVerts(mo);
  135. }
  136. void tnsAddMMeshPlane(tnsMeshObject* mo, real size){
  137. tnsMVert *mv1,*mv2,*mv3,*mv4;
  138. mv1=tnsMMeshNewVert(mo); mv1->p[0]=size; mv1->p[1]=size; mv1->flags|=TNS_MESH_FLAG_SELECTED;
  139. mv2=tnsMMeshNewVert(mo); mv2->p[0]=-size; mv2->p[1]=size; mv2->flags|=TNS_MESH_FLAG_SELECTED;
  140. mv3=tnsMMeshNewVert(mo); mv3->p[0]=-size; mv3->p[1]=-size; mv3->flags|=TNS_MESH_FLAG_SELECTED;
  141. mv4=tnsMMeshNewVert(mo); mv4->p[0]=size; mv4->p[1]=-size; mv4->flags|=TNS_MESH_FLAG_SELECTED;
  142. tnsMMeshMakeFace4v(mo,mv1,mv2,mv3,mv4);
  143. }
  144. void tnsTrangulateFaceSimple(tnsMeshObject* mo, int fi, tnsFace* f, int* ebuf){
  145. for(int i=0;i<f->looplen-2;i++){
  146. ebuf[i*3]=f->loop[i];
  147. ebuf[i*3+1]=f->loop[i+1];
  148. ebuf[i*3+2]=fi+mo->totv; //f->loop[i+2];
  149. }
  150. }
  151. void tnsTrangulateFaceSimpleM(tnsMeshObject* mo, int fi, tnsMFace* mf, int* ebuf){
  152. tnsMVert* mv=0,*mvs; int i=0;
  153. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  154. laListItemPointer* next=lip->pNext; if(!next) next=mf->l.pFirst; tnsMEdge* me0=lip->p, *me1=next->p;
  155. if(next==mf->l.pLast){ break; }
  156. mvs=tnsMMeshEdgeStartingVert(me0,me1);
  157. if(!mv) mv=mvs;
  158. ebuf[i*3]=mvs->i; tnsMVert*mm=tnsMMeshEdgeAnotherVert(me0,mvs);
  159. ebuf[i*3+1]=mm->i;
  160. ebuf[i*3+2]=fi+mo->totmv; /*tnsMMeshEdgeAnotherVert(me1,mm)->i;*/ i++;
  161. }
  162. }
  163. tnsMVert* tnsGetMFaceLastVert(tnsMFace* mf){
  164. laListItemPointer* lip=mf->l.pLast; laListItemPointer* next=mf->l.pFirst; tnsMEdge* me0=lip->p, *me1=next->p;
  165. return tnsMMeshEdgeStartingVert(me0,me1);
  166. }
  167. int* tnsGetTriangulatedBatch(tnsMeshObject* mo, int* totelem){
  168. int tottri=0;
  169. if(mo->Mode==TNS_MESH_OBJECT_MODE){ for(int i=0;i<mo->totf;i++){ tottri+=(mo->f[i].looplen-2); } }
  170. else{ for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ tottri+=mf->looplen-2; } }
  171. if(!tottri) return 0;
  172. int* ebuf=calloc(1,sizeof(int)*tottri*3); int* pebuf=ebuf;
  173. if(mo->Mode==TNS_MESH_OBJECT_MODE){ for(int i=0;i<mo->totf;i++){ tnsTrangulateFaceSimple(mo, i, &mo->f[i], pebuf); pebuf+=(mo->f[i].looplen-2)*3; } }
  174. else{ int i=0; for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ tnsTrangulateFaceSimpleM(mo, i, mf, pebuf); pebuf+=(mf->looplen-2)*3; i++; } }
  175. *totelem = tottri;
  176. return ebuf;
  177. }
  178. float* tnsGetDrawingVertArray(tnsMeshObject* mo, int* r_tot_render_v, float** r_normals, float** idcolors, float** editcolors, int** edgeelems, int* r_tote){
  179. if(!mo->totv&&!mo->totmv) return 0;
  180. int objmode=mo->Mode==TNS_MESH_OBJECT_MODE;
  181. int totv=objmode?mo->totv:mo->totmv;
  182. int tote=objmode?mo->tote:mo->totme; *r_tote=tote;
  183. int totf=objmode?mo->totf:mo->totmf;
  184. int extraverts=objmode?mo->totf:(mo->totme*2+mo->totmf);
  185. float* p=calloc(1,(totv+extraverts)*3*sizeof(float)); *r_tot_render_v=(totv+extraverts);
  186. float* n=calloc(1,(totv+extraverts)*3*sizeof(float)); *r_normals = n;
  187. if(objmode){
  188. for(int i=0;i<totv;i++){ tnsVectorSet3v(&p[i*3],mo->v[i].p); tnsVectorSet3v(&n[i*3],mo->v[i].n); }
  189. int start=mo->totv*3; for(int i=0;i<totf;i++){
  190. tnsVectorSet3v(&p[start+i*3],mo->v[mo->f[i].loop[mo->f[i].looplen-1]].p); tnsVectorSet3v(&n[start+i*3],mo->f[i].n);
  191. }
  192. }else{
  193. (*idcolors)=calloc(1,(totv+extraverts)*3*sizeof(float));
  194. if(extraverts){
  195. (*edgeelems)=calloc(1,(extraverts)*2*sizeof(int));
  196. (*editcolors)=calloc(1,totv*4*sizeof(float)*extraverts);
  197. }
  198. for(tnsMVert*mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ int i=mv->i;
  199. tnsVectorSet3v(&p[i*3],mv->p); tnsVectorSet3v(&n[i*3],mv->n);
  200. int id=(i+1); real r=(real)((id & 0x000000FF)>>0)/255.0; real g=(real)((id & 0x0000FF00)>>8)/255.0; real b=(real)((id & 0x00FF0000)>>16)/255.0;
  201. (*idcolors)[i*3]=r; (*idcolors)[i*3+1]=g; (*idcolors)[i*3+2]=b;
  202. real* c=(mv->flags&TNS_MESH_FLAG_SELECTED)?laAccentColor(LA_BT_SVERTEX):laAccentColor(LA_BT_VERTEX);
  203. if(extraverts){ (*editcolors)[i*4]=c[0]; (*editcolors)[i*4+1]=c[1]; (*editcolors)[i*4+2]=c[2]; (*editcolors)[i*4+3]=c[3];}
  204. }
  205. if(extraverts){
  206. for(tnsMEdge*me=mo->me.pFirst;me;me=me->Item.pNext){ int ei=me->i;
  207. (*edgeelems)[ei*2]=mo->totmv+mo->totmf+ei*2; (*edgeelems)[ei*2+1]=mo->totmv+mo->totmf+ei*2+1;
  208. float* eidcolor1=&(*idcolors)[(*edgeelems)[ei*2]*3], *eidcolor2=&(*idcolors)[(*edgeelems)[ei*2+1]*3];
  209. int id=((ei+1)|TNS_MMESH_EDGE_BIT); real r=(real)((id & 0x000000FF)>>0)/255.0; real g=(real)((id & 0x0000FF00)>>8)/255.0; real b=(real)((id & 0x00FF0000)>>16)/255.0;
  210. tnsVectorSet3(eidcolor1,r,g,b); tnsVectorSet3(eidcolor2,r,g,b);
  211. int se1=(*edgeelems)[ei*2]; tnsVectorSet3v(&p[se1*3],me->vl->p);
  212. int se2=(*edgeelems)[ei*2+1]; tnsVectorSet3v(&p[se2*3],me->vr->p);
  213. float* eedcolor1=&(*editcolors)[(*edgeelems)[ei*2]*4], *eedcolor2=&(*editcolors)[(*edgeelems)[ei*2+1]*4];
  214. real* c=(me->flags&TNS_MESH_FLAG_SELECTED)?laAccentColor(LA_BT_SEDGE):laAccentColor(LA_BT_EDGE);
  215. tnsVectorSet4v(eedcolor1,c); tnsVectorSet4v(eedcolor2,c);
  216. }
  217. }
  218. int start=mo->totmv*3; int fi=0; for(tnsMFace*mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){
  219. tnsMVert* sv=tnsGetMFaceLastVert(mf); tnsVectorSet3v(&p[start+fi*3],sv->p); tnsVectorSet3v(&n[start+fi*3],mf->n); fi++;
  220. }
  221. }
  222. return p;
  223. }
  224. int* tnsGetEdgeBatch(tnsMeshObject* mo){
  225. if(!mo->totme) return 0;
  226. int* ebuf=calloc(1,sizeof(int)*mo->totme*2); int* pebuf=ebuf; int i=0;
  227. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ ebuf[i*2]=me->vl->i; ebuf[i*2+1]=me->vr->i; i++; }
  228. return ebuf;
  229. }
  230. void tnsInvalidateMeshBatch(tnsMeshObject* mo){
  231. if(mo->Base.Type!=TNS_OBJECT_MESH) return;
  232. if(mo->Batch) tnsDeleteBatch(mo->Batch); mo->Batch=0;
  233. if(mo->Base.PlayDuplicate){ tnsMeshObject*pmo=mo->Base.PlayDuplicate; pmo->Batch=0; }
  234. }
  235. void tnsRegenerateMeshBatch(tnsMeshObject* mo){
  236. if(!mo) return;
  237. if(mo->Batch) tnsDeleteBatch(mo->Batch); mo->Batch=0;
  238. real meshcolor[4]={0.8,0.8,0.8,1};
  239. int tottri; int* elem = tnsGetTriangulatedBatch(mo, &tottri);
  240. float* idcolors=0,*editcolors=0; int docolors=mo->Mode==TNS_MESH_EDIT_MODE;
  241. int totv,tote; int* eelems=0; float* n=0;
  242. float* v = tnsGetDrawingVertArray(mo,&totv,&n,&idcolors,&editcolors,&eelems,&tote);
  243. if(!v){ if(elem){free(elem);} if(eelems){free(eelems);} return; }
  244. mo->Batch = tnsCreateBatch(totv, 3, v, 3, n, 4, editcolors);
  245. tnsBatchCommand*c;
  246. if(elem){
  247. c=tnsCreateCommand(mo->Batch, "body", tottri, 3, GL_TRIANGLES, elem, 0);
  248. tnsCommandUseUniformColor(c,meshcolor);
  249. }
  250. free(elem); free(v); free(n);
  251. if(mo->Mode==TNS_MESH_EDIT_MODE){
  252. elem=tnsGetEdgeBatch(mo); if(elem) {
  253. c= tnsCreateCommand(mo->Batch, "lines", mo->totme, 2, GL_LINES, elem, 1); free(elem);
  254. //tnsCommandUseUniformColor(c, laAccentColor(LA_BT_EDGE));
  255. }
  256. c= tnsCreateCommand(mo->Batch, "verts", mo->totmv, 3, GL_POINTS, 0, 1);
  257. c= tnsCreateCommand(mo->Batch, "verts_select", mo->totmv, 3, GL_POINTS, 0, 1);
  258. tnsCommandOverrideColorArray(c, mo->Batch->NumVert, 3, idcolors);
  259. if(eelems){
  260. c= tnsCreateCommand(mo->Batch, "edges_select", mo->totme, 2, GL_LINES, eelems, 1);
  261. tnsCommandOverrideColorArray(c, mo->Batch->NumVert, 3, idcolors);
  262. c= tnsCreateCommand(mo->Batch, "edges", mo->totme, 2, GL_LINES, eelems, 1);
  263. tnsCommandOverrideColorArray(c, mo->Batch->NumVert, 4, editcolors);
  264. }
  265. //for(int i=0;i<mo->totme*2;i++){ printf("%d ",eelems[i]); } printf("\n");
  266. }
  267. if(idcolors) free(idcolors); if(editcolors) free(editcolors); if(eelems) free(eelems);
  268. if(mo->Base.PlayDuplicate){ tnsMeshObject*pmo=mo->Base.PlayDuplicate; pmo->Batch=mo->Batch; }
  269. }
  270. void tnsEnsureMeshBatch(tnsMeshObject* mo){
  271. if(mo->Base.Type!=TNS_OBJECT_MESH) return;
  272. if(mo->Batch) return;
  273. tnsRegenerateMeshBatch(mo);
  274. }
  275. void tnsDrawMeshObjectEdit(tnsMeshObject* mo, int MeshSelectionType){
  276. tnsUniformUseOffset(T->immShader,1);
  277. glPointSize(6); glLineWidth(3);
  278. if(MeshSelectionType==LA_CANVAS_SELECT_MODE_VERTS){
  279. tnsDrawBatch(mo->Batch,"verts",0,0); tnsDrawBatch(mo->Batch,"lines",0,0);
  280. }else{
  281. tnsDrawBatch(mo->Batch,"edges",0,0);
  282. }
  283. glPointSize(1); glLineWidth(1);
  284. tnsUniformUseOffset(T->immShader,0);
  285. }
  286. void tnsDrawMeshObjectOverlay(tnsEvaluatedInstance* ei, la3DObjectDrawExtra* de){
  287. tnsEnsureMeshBatch(ei->Object);
  288. tnsDrawMeshObjectEdit(ei->Object, de?de->MeshEditType:LA_CANVAS_SELECT_MODE_VERTS);
  289. if(((tnsMeshObject*)ei->Object)->ExtraBatch){ tnsDrawBatch(((tnsMeshObject*)ei->Object)->ExtraBatch,0,0,0); }
  290. }
  291. void tnsDrawMeshObjectSelectionID(tnsEvaluatedInstance* ei, void* unused){
  292. tnsEnsureMeshBatch(ei->Object);
  293. int i=ei->InstanceSelectionID; real color[4]={0,0,0,1}; TNS_ID_TO_COLOR(color,i);
  294. tnsDrawBatch(((tnsMeshObject*)ei->Object)->Batch,"body",color,0);
  295. }
  296. void tnsDrawMeshObjectOutline(tnsEvaluatedInstance* ei, void* unused){
  297. real* color=(ei->IsActive)?laAccentColor(LA_BT_TEXT):laAccentColor(LA_BT_NORMAL);
  298. tnsDrawBatch(((tnsMeshObject*)ei->Object)->Batch, "body", color, 0);
  299. }
  300. void tnsDrawMeshObject(tnsEvaluatedInstance* ei, void* unused){
  301. tnsEnsureMeshBatch(ei->Object);
  302. tnsUseNormal(1); tnsDrawBatch(((tnsMeshObject*)ei->Object)->Batch,"body",0,0); tnsUseNormal(0);
  303. }
  304. void tnsEvaluateMeshObject(tnsMeshObject* mo, tnsEvaluateData* ed){
  305. tnsEnsureMeshBatch(mo);
  306. tnsAddEvaluatedInstance(ed,mo,tnsDrawMeshObject,TNS_EVAL_LAYER_SOLID,0,0,0);
  307. if(ed->FillOutline && (!ed->OverrideID)){
  308. if((mo->Base.Flags&TNS_OBJECT_FLAGS_SELECTED) && (mo->Mode!=TNS_MESH_EDIT_MODE)){
  309. tnsAddEvaluatedInstance(ed,mo,tnsDrawMeshObjectOutline,TNS_EVAL_LAYER_OUTLINE,ed->Active==mo,0,0);
  310. }
  311. }
  312. if(ed->FillSelectionID){
  313. tnsAddEvaluatedInstance(ed,mo,tnsDrawMeshObjectSelectionID,TNS_EVAL_LAYER_SELECTION,0,1,mo->Base.SelectID);
  314. }
  315. if(mo->Mode==TNS_MESH_EDIT_MODE && (!ed->OverrideID)){
  316. tnsAddEvaluatedInstance(ed,mo,tnsDrawMeshObjectOverlay,TNS_EVAL_LAYER_OVERLAY,0,0,0);
  317. }
  318. }
  319. tnsMFace* tnsMMeshNewFace(tnsMeshObject* mo){ tnsMFace* mf=memAcquireSimple(sizeof(tnsMFace)); mf->i=mo->totmf; mo->totmf++; lstAppendItem(&mo->mf,mf); return mf; }
  320. tnsMEdge* tnsMMeshNewEdge(tnsMeshObject* mo){ tnsMEdge* me=memAcquireSimple(sizeof(tnsMEdge)); me->i=mo->totme; mo->totme++; lstAppendItem(&mo->me,me); return me; }
  321. tnsMVert* tnsMMeshNewVert(tnsMeshObject* mo){ tnsMVert* mv=memAcquireSimple(sizeof(tnsMVert)); mv->i=mo->totmv; mo->totmv++; lstAppendItem(&mo->mv,mv); return mv; }
  322. void tnsMMeshCalculateNormalFrom(tnsMFace* mf){
  323. tnsVector3d accum={0}; tnsVector3d d0,d1; tnsVector3d naccum={0},nn={0};if(mf->flags&TNS_MESH_FLAG_PICKED) return;
  324. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  325. laListItemPointer* next=lip->pNext; if(!next) next=mf->l.pFirst; tnsMEdge* me0=lip->p, *me1=next->p;
  326. tnsMVert* v0=tnsMMeshEdgeStartingVert(me0,me1); tnsMVert* v1=tnsMMeshEdgeShareVert(me0,me1); tnsMVert* v2=tnsMMeshEdgeAnotherVert(me1, v1);
  327. tnsVectorAccum3d(accum, v0->p); tnsVectorAccum3d(accum, v1->p); tnsVectorAccum3d(accum, v2->p);
  328. tnsVectorMinus3d(d0,v0->p,v1->p); tnsVectorMinus3d(d1,v2->p,v1->p); real len=tnsVectorCross3d(nn,d0,d1); tnsVectorMultiSelf3d(nn, len);
  329. tnsVectorAccum3d(naccum, nn);
  330. //if(v0==me0->vr){ me0->flags|=TNS_MESH_FLAG_LOOP_REVERSE; } // TODO: consistency not implemented.
  331. }
  332. tnsNormalize3d(mf->n, naccum); tnsVectorMulti3d(mf->c, accum, 1.0/(mf->looplen+2));
  333. mf->flags|=TNS_MESH_FLAG_PICKED;
  334. if(mf->flags&TNS_MESH_FLAG_SELECTED){
  335. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  336. tnsMEdge* me=lip->p; tnsMFace* of=((mf==me->fl)?me->fr:me->fl);
  337. if(of) tnsMMeshCalculateNormalFrom(of);
  338. }
  339. }
  340. }
  341. void tnsMMeshCalculateFaceVertNormal(tnsMFace* mf){
  342. tnsVector3d accum={0};
  343. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  344. laListItemPointer* next=lip->pNext; if(!next) next=mf->l.pFirst; tnsMEdge* me0=lip->p, *me1=next->p;
  345. tnsMVert* mv=tnsMMeshEdgeStartingVert(me0,me1); int fcount=0;
  346. for(laListItemPointer* el=mv->elink.pFirst;el;el=el->pNext){ tnsMEdge* ve=lip->p;
  347. if(ve->fl){ tnsVectorAccum3d(accum, ve->fl->n); fcount++; }
  348. if(ve->fr){ tnsVectorAccum3d(accum, ve->fr->n); fcount++; }
  349. }
  350. if(!fcount) accum[2]=1; else tnsNormalizeSelf3d(accum);
  351. tnsVectorSet3v(mv->n, accum);
  352. }
  353. }
  354. int tnsMMeshCalculateNormal(tnsMeshObject* mo){
  355. tnsMMeshClearExtraFlags(mo); int ran=0;
  356. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ if(!(mv->flags&TNS_MESH_FLAG_SELECTED)) continue;
  357. for(laListItemPointer* lip=mv->elink.pFirst; lip;lip=lip->pNext){
  358. tnsMEdge*me=lip->p; if((!me->fl) && (!me->fr)) continue;
  359. if(me->fl) tnsMMeshCalculateNormalFrom(me->fl); if(me->fr) tnsMMeshCalculateNormalFrom(me->fr); ran=1;
  360. }
  361. }
  362. //for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ if((!(mf->flags&TNS_MESH_FLAG_SELECTED))||(mf->flags&TNS_MESH_FLAG_PICKED)) continue;
  363. // tnsMMeshCalculateNormalFrom(mf); ran=1;
  364. //}
  365. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ if(!(mf->flags&TNS_MESH_FLAG_PICKED)) continue;
  366. tnsMMeshCalculateFaceVertNormal(mf); ran=1;
  367. }
  368. return ran;
  369. }
  370. void tnsMMeshEdgeAssignVerts(tnsMEdge* me,tnsMVert* mv1,tnsMVert* mv2){
  371. if(me->vl||me->vr){ return; } //if((me->vl==mv1&&me->vr=mv2) || (me->vl==mv2&&me->vr=mv1))
  372. me->vl=mv1; me->vr=mv2;
  373. lstAppendPointer(&me->vl->elink,me); lstAppendPointer(&me->vr->elink,me);
  374. }
  375. tnsMEdge* tnsMMeshVertsShareEdge(tnsMVert* mv0, tnsMVert* mv1){
  376. for(laListItemPointer*lip=mv0->elink.pFirst;lip;lip=lip->pNext){ tnsMEdge* me=lip->p; if(tnsMMeshEdgeAnotherVert(me, mv0)==mv1) return me; } return 0;
  377. }
  378. int tnsMMeshEdgeHasVert(tnsMEdge* me, tnsMVert* mv){
  379. if(me->vl==mv || me->vr==mv) return 1; return 0;
  380. }
  381. tnsMFace* tnsMMeshEdgeShareFace(tnsMEdge* me0, tnsMEdge* me1){
  382. if(me0->fl==me1->fl || me0->fl==me1->fr) return me0->fl;
  383. if(me0->fr==me1->fl || me0->fr==me1->fr) return me0->fr;
  384. return 0;
  385. }
  386. tnsMVert* tnsMMeshEdgeShareVert(tnsMEdge* me0, tnsMEdge* me1){
  387. if(me0->vl==me1->vl || me0->vl==me1->vr) return me0->vl;
  388. if(me0->vr==me1->vl || me0->vr==me1->vr) return me0->vr;
  389. return 0;
  390. }
  391. tnsMVert* tnsMMeshEdgeAnotherVert(tnsMEdge* me, tnsMVert* v){
  392. if(me->vl==v) return me->vr; if(me->vr==v) return me->vl;
  393. return 0;
  394. }
  395. tnsMVert* tnsMMeshEdgeStartingVert(tnsMEdge* me0, tnsMEdge* me1){
  396. tnsMVert* sv=tnsMMeshEdgeShareVert(me0,me1); if(!sv) return 0;
  397. return tnsMMeshEdgeAnotherVert(me0, sv);
  398. }
  399. void tnsMMeshFaceAddEdge(tnsMFace* mf, tnsMEdge* me){
  400. lstAppendPointer(&mf->l, me); mf->looplen++;
  401. if(!me->fl) me->fl=mf; elif(!me->fr) me->fr=mf;
  402. }
  403. void tnsMMeshFaceAddEdgeReplaceFace(tnsMFace* mf, tnsMEdge* me, tnsMFace* to_be_replaced){
  404. lstAppendPointer(&mf->l, me); mf->looplen++;
  405. if(me->fl==to_be_replaced) me->fl=mf; elif(me->fr==to_be_replaced) me->fr=mf;
  406. elif(!me->fl) me->fl=mf; elif(!me->fr) me->fr=mf;
  407. }
  408. tnsMFace* tnsMMeshFaceHasVert(tnsMFace* mf, tnsMVert* mv){
  409. if(!mf||!mv) return 0; for(laListItemPointer*lip=mf->l.pFirst;lip;lip=lip->pNext){ if(tnsMMeshEdgeAnotherVert(lip->p, mv)) return mf; }
  410. return 0;
  411. }
  412. tnsMFace* tnsMMeshVertsShareFace(tnsMVert* v1, tnsMVert* v2){
  413. tnsMFace* mf=0; for(laListItemPointer*lip=v1->elink.pFirst;lip;lip=lip->pNext){
  414. tnsMEdge* me=lip->p; if((mf=tnsMMeshFaceHasVert(me->fl, v2)) || (mf=tnsMMeshFaceHasVert(me->fr, v2))) return mf;
  415. } return 0;
  416. }
  417. int tnsMMeshSplitFace(tnsMeshObject* mo, tnsMFace* mf, tnsMEdge* me, tnsMFace** r_f1, tnsMFace** r_f2){
  418. tnsMEdge* NextE; laListItemPointer* NextLip, *StartLip=0, *EndLip=0; int guard=0;
  419. for(laListItemPointer*lip=mf->l.pFirst;lip;lip=NextLip){
  420. NextLip=lip->pNext?lip->pNext:mf->l.pFirst; NextE=NextLip->p;
  421. if(tnsMMeshEdgeHasVert(me,tnsMMeshEdgeShareVert(NextE,lip->p))){
  422. if(!StartLip) StartLip=NextLip;
  423. else{EndLip=NextLip; break;} }
  424. guard++; if(guard>mf->looplen) return 0; // me is not across mf.
  425. }
  426. tnsMFace* f1=tnsMMeshNewFace(mo);
  427. for(laListItemPointer*lip=StartLip;lip;lip=NextLip){ NextLip=lip->pNext?lip->pNext:mf->l.pFirst;
  428. if(lip==EndLip){ tnsMMeshFaceAddEdgeReplaceFace(f1, me, mf); break; } tnsMMeshFaceAddEdgeReplaceFace(f1, lip->p, mf);
  429. }
  430. tnsMFace* f2=tnsMMeshNewFace(mo);
  431. for(laListItemPointer*lip=EndLip;lip;lip=NextLip){ NextLip=lip->pNext?lip->pNext:mf->l.pFirst;
  432. if(lip==StartLip){ tnsMMeshFaceAddEdgeReplaceFace(f2, me, mf); break; } tnsMMeshFaceAddEdgeReplaceFace(f2, lip->p, mf);
  433. }
  434. tnsMMeshRemoveFaceOnly(mo, mf);
  435. if(r_f1){ *r_f1=f1; } if(r_f2){ *r_f2=f2; }
  436. //tnsPrintMeshTopology(mo);
  437. return 1;
  438. }
  439. tnsMEdge* tnsMMeshMakeEdge(tnsMeshObject* mo, tnsMVert* v1, tnsMVert* v2){
  440. for(laListItemPointer*lip=v1->elink.pFirst;lip;lip=lip->pNext){ if(tnsMMeshEdgeAnotherVert(lip->p, v1)==v2) return lip->p; }
  441. // for(laListItemPointer*lip=v2->elink.pFirst;lip;lip=lip->pNext){ if(tnsMMeshEdgeAnotherVert(lip->p, v2)==v1) return lip->p; } shouldn't need.
  442. tnsMFace* mf=tnsMMeshVertsShareFace(v1,v2);
  443. tnsMEdge* me=tnsMMeshNewEdge(mo); tnsMMeshEdgeAssignVerts(me, v1, v2);
  444. if(mf){ tnsMMeshSplitFace(mo, mf, me, 0,0); }
  445. return me;
  446. }
  447. int tnsMMeshFaceMatchesN(tnsMFace* mf, int ecount, laListHandle* eip){
  448. if(!mf||mf->looplen!=ecount) return 0; laListItemPointer* lipe=eip->pFirst;
  449. for(int i=0;i<ecount;i++){
  450. tnsMEdge* me=lipe->p; int found=0; lipe=lipe->pNext;
  451. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  452. if(lip->p==me){ found=1; break; }
  453. }
  454. if(!found){ return 0; }
  455. }
  456. return 1;
  457. }
  458. int tnsMMeshFaceMatches(tnsMFace* mf, int ecount, ...){
  459. if(!mf||mf->looplen!=ecount) return 0;
  460. va_list list; va_start(list, ecount);
  461. for(int i=0;i<ecount;i++){
  462. tnsMEdge* me=va_arg(list, tnsMEdge*); int found=0;
  463. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  464. if(lip->p==me){ found=1; break; }
  465. }
  466. if(!found){ va_end(list); return 0; }
  467. }
  468. va_end(list); return 1;
  469. }
  470. tnsMFace* tnsMMeshMakeFaceN(tnsMeshObject* mo, int count, laListHandle* vip, tnsMEdge** r_fallback_me){
  471. if(count<2){ return 0; }
  472. if(count==2){ tnsMEdge* me=tnsMMeshMakeEdge(mo,((laListItemPointer*)vip->pFirst)->p,((laListItemPointer*)vip->pLast)->p); if(r_fallback_me)*r_fallback_me=me; return 0; }
  473. laListHandle el={0};
  474. for(laListItemPointer*lip=vip->pFirst;lip;lip=lip->pNext){
  475. tnsMVert* mv=lip->p; laListItemPointer*nlip=lip->pNext?((laListItemPointer*)lip->pNext):vip->pFirst; tnsMVert* nextmv=nlip->p;
  476. lstAppendPointer(&el,tnsMMeshMakeEdge(mo,mv,nextmv));
  477. }
  478. tnsMEdge* e1=((laListItemPointer*)el.pFirst)->p;
  479. if(tnsMMeshFaceMatchesN(e1->fl, count, &el))return e1->fl; if(tnsMMeshFaceMatchesN(e1->fr, count, &el))return e1->fr;
  480. for(laListItemPointer* lip=el.pFirst;lip;lip=lip->pNext){ tnsMEdge*me=lip->p; if(me->fl&&me->fr) return 0; }
  481. tnsMFace* mf=tnsMMeshNewFace(mo);
  482. for(laListItemPointer* lip=el.pFirst;lip;lip=lip->pNext){ tnsMEdge*me=lip->p; tnsMMeshFaceAddEdge(mf, me); }
  483. while(lstPopPointer(&el));
  484. return mf;
  485. }
  486. tnsMFace* tnsMMeshMakeFace4v(tnsMeshObject* mo, tnsMVert* v1,tnsMVert* v2,tnsMVert* v3,tnsMVert* v4){
  487. tnsMEdge* e1=tnsMMeshMakeEdge(mo,v1,v2); tnsMEdge* e2=tnsMMeshMakeEdge(mo,v2,v3);
  488. tnsMEdge* e3=tnsMMeshMakeEdge(mo,v3,v4); tnsMEdge* e4=tnsMMeshMakeEdge(mo,v4,v1);
  489. if(tnsMMeshFaceMatches(e1->fl,4,e1,e2,e3,e4)) return e1->fl; if(tnsMMeshFaceMatches(e1->fr,4,e1,e2,e3,e4)) return e1->fr; //should not need more
  490. if((e1->fl&&e1->fr) || (e2->fl&&e2->fr) || (e3->fl&&e3->fr) || (e4->fl&&e4->fr)) return 0;
  491. tnsMFace* mf=tnsMMeshNewFace(mo);
  492. tnsMMeshFaceAddEdge(mf,e1); tnsMMeshFaceAddEdge(mf,e2);
  493. tnsMMeshFaceAddEdge(mf,e3); tnsMMeshFaceAddEdge(mf,e4);
  494. return mf;
  495. }
  496. int tnsMMeshLoopIsInverted(laListItemPointer* l){
  497. tnsMEdge* me=l->p; if(l->pNext){ tnsMEdge*next=((laListItemPointer*)l->pNext)->p; if(me->vr==tnsMMeshEdgeShareVert(me,next))return 0; return 1; }
  498. else{ tnsMEdge*prev=((laListItemPointer*)l->pPrev)->p; if(me->vl==tnsMMeshEdgeShareVert(prev,me))return 0; return 1; }
  499. }
  500. int tnsMMeshEdgeInsertVert(tnsMeshObject* mo, tnsMEdge* me, tnsMVert* mv, tnsMVert* ref_e1v_optional, tnsMEdge** r_e1, tnsMEdge** r_e2){
  501. if(mv->elink.pFirst||mv->elink.pLast||me->vl==mv||me->vl==mv) return 0;
  502. tnsMEdge* me1=tnsMMeshNewEdge(mo),*me2=tnsMMeshNewEdge(mo);
  503. me1->fl=me2->fl=me->fl; me1->fr=me2->fr=me->fr;
  504. tnsMMeshEdgeAssignVerts(me1,me->vl,mv); tnsMMeshEdgeAssignVerts(me2,mv,me->vr);
  505. laListItemPointer* lipa1=memAcquireSimple(sizeof(laListItemPointer)),*lipa2=memAcquireSimple(sizeof(laListItemPointer)); lipa1->p=me1; lipa2->p=me2;
  506. laListItemPointer* lipb1=memAcquireSimple(sizeof(laListItemPointer)),*lipb2=memAcquireSimple(sizeof(laListItemPointer)); lipb1->p=me1; lipb2->p=me2;
  507. if(me->fl) for(laListItemPointer* lip=me->fl->l.pFirst;lip;lip=lip->pNext){
  508. tnsMEdge*ie=lip->p; if(ie!=me) continue; if(tnsMMeshLoopIsInverted(lip)){ LA_SWAP(laListItemPointer*,lipa1,lipa2); }
  509. lstInsertItemBefore(&me->fl->l,lipa1,lip); lstInsertItemBefore(&me->fl->l,lipa2,lip); lstRemoveItem(&me->fl->l, lip); memLeave(lip); me->fl->looplen++;
  510. }
  511. if(me->fr) for(laListItemPointer* lip=me->fr->l.pFirst;lip;lip=lip->pNext){
  512. tnsMEdge*ie=lip->p; if(ie!=me) continue; if(tnsMMeshLoopIsInverted(lip)){ LA_SWAP(laListItemPointer*,lipb1,lipb2); }
  513. lstInsertItemBefore(&me->fr->l,lipb1,lip); lstInsertItemBefore(&me->fr->l,lipb2,lip); lstRemoveItem(&me->fr->l, lip); memLeave(lip); me->fr->looplen++;
  514. }
  515. me->fl=me->fr=0; tnsMMeshRemoveEdgeFace(mo,me);
  516. if(ref_e1v_optional&&tnsMMeshEdgeShareVert(me1, ref_e1v_optional)){ if(r_e1)*r_e1=me1; if(r_e2)*r_e2=me2; }else{ if(r_e1)*r_e1=me2; if(r_e2)*r_e2=me1; }
  517. //tnsPrintMeshTopology(mo);
  518. return 1;
  519. }
  520. tnsMVert* tnsMMeshEdgeInsertVertAt(tnsMeshObject* mo, tnsMEdge* me, real at, tnsMVert* ref_e1v_optional, tnsMEdge** r_e1, tnsMEdge** r_e2){
  521. if(!me||at<=0||at>=1) return 0; tnsMVert* mv=tnsMMeshNewVert(mo);
  522. tnsInterpolate3dv(me->vl->p, me->vr->p, at, mv->p);
  523. if(tnsMMeshEdgeInsertVert(mo,me,mv,ref_e1v_optional,r_e1,r_e2)) return mv; return 0;
  524. }
  525. void tnsMMeshRemoveFaceOnly(tnsMeshObject* mo, tnsMFace* mf){
  526. if(!mf) return; tnsMEdge* me;
  527. while(me=lstPopPointerLeave(&mf->l)){ if(me->fl==mf) me->fl=0; elif(me->fr==mf) me->fr=0; }
  528. lstRemoveItem(&mo->mf,mf); memLeave(mf); mo->totmf--;
  529. tnsMMeshClearFirstLastSelection(mo);
  530. }
  531. void tnsMMeshRemoveEdgeFace(tnsMeshObject* mo, tnsMEdge* me){
  532. if(!me) return;
  533. tnsMMeshRemoveFaceOnly(mo, me->fl); tnsMMeshRemoveFaceOnly(mo, me->fr);
  534. lstRemovePointerLeave(&me->vl->elink, me); lstRemovePointerLeave(&me->vr->elink, me);
  535. lstRemoveItem(&mo->me,me); memLeave(me); mo->totme--;
  536. tnsMMeshClearFirstLastSelection(mo);
  537. }
  538. void tnsMMeshRemoveVertEdgeFace(tnsMeshObject* mo, tnsMVert* mv){
  539. if(!mv) return; tnsMEdge* me;
  540. while(me=lstPopPointerLeave(&mv->elink)){ tnsMMeshRemoveEdgeFace(mo,me); }
  541. lstRemoveItem(&mo->mv,mv); memLeave(mv); mo->totmv--;
  542. tnsMMeshClearFirstLastSelection(mo);
  543. }
  544. void tnsMMeshRefreshIndex(tnsMeshObject* mo){
  545. int i;
  546. i=0; for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ mv->i=i; i++; } mo->totmv=i;
  547. i=0; for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ me->i=i; i++; } mo->totme=i;
  548. i=0; for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ mf->i=i; i++; } mo->totmf=i;
  549. }
  550. void tnsClearMesh(tnsMeshObject* mo){
  551. arrFree(&mo->v, &mo->maxv); mo->totv=0;
  552. arrFree(&mo->e, &mo->maxe); mo->tote=0;
  553. if(mo->f) for(int i=0;i<mo->totf;i++){ if(mo->f[i].loop) free(mo->f[i].loop); }
  554. arrFree(&mo->f, &mo->maxf); mo->totf=0;
  555. }
  556. void tnsClearMMesh(tnsMeshObject* mo){
  557. tnsMFace* mf; tnsMEdge* me; tnsMVert* mv;
  558. while(mf=lstPopItem(&mo->mf)){ while(lstPopPointerLeave(&mf->l)); memLeave(mf); }
  559. while(me=lstPopItem(&mo->me)){ memLeave(me); }
  560. while(mv=lstPopItem(&mo->mv)){ while(lstPopPointerLeave(&mv->elink)); memLeave(mv); }
  561. mo->totmv=mo->totme=mo->totmf=0;
  562. }
  563. void tnsMMeshFromMesh(tnsMeshObject* mo){
  564. tnsClearMMesh(mo);
  565. tnsEdgeHash* eh=tnsCreateEdgeHash(mo->totv); //mo->totmv=mo->totv; mo->totme=mo->tote; mo->totmf=mo->totf;
  566. for(int i=0;i<mo->totf;i++){
  567. tnsFace* f=&mo->f[i];
  568. for(int j=0;j<f->looplen-1;j++){ tnsEdgeHashAddVertPair(eh, f->loop[j], f->loop[j+1]); }
  569. tnsEdgeHashAddVertPair(eh, f->loop[f->looplen-1], f->loop[0]);
  570. }
  571. for(int i=0;i<eh->max;i++){
  572. tnsEdgeHashVert* ehv=&eh->vl[i];
  573. for(int j=0;j<ehv->next;j++){ tnsMEdge*me=tnsMMeshNewEdge(mo); ehv->e[j].me=me; }
  574. }
  575. for(int i=0;i<mo->totv;i++){ tnsVert*v=&mo->v[i]; tnsMVert*mv=tnsMMeshNewVert(mo); eh->vl[i].mv=mv;
  576. tnsVectorSet3v(mv->p,mo->v[i].p); mv->flags=mo->v[i].flags; tnsVectorSet3v(mv->n,v->n); }
  577. for(int i=0;i<mo->totf;i++){
  578. tnsFace* f=&mo->f[i]; tnsMFace* mf=tnsMMeshNewFace(mo); mf->flags=f->flags;
  579. for(int j=0;j<f->looplen;j++){ int v2=j+1; if(j==f->looplen-1) v2=0;
  580. tnsEdgeHashEdge* ehe=tnsEdgeHashGetEdge(eh,f->loop[j],f->loop[v2]);
  581. tnsMEdge* me=ehe->me; tnsMMeshEdgeAssignVerts(me,eh->vl[f->loop[j]].mv,eh->vl[f->loop[v2]].mv);
  582. tnsMMeshFaceAddEdge(mf,me); tnsVectorSet3v(mf->n,f->n);
  583. }
  584. }
  585. tnsMMeshEnsureSelectionFromVerts(mo);
  586. tnsDestroyEdgeHash(eh);
  587. }
  588. void tnsMeshFromMMesh(tnsMeshObject* mo){
  589. tnsClearMesh(mo);
  590. tnsInitMesh(mo, mo->totmv, 0, mo->totmf); int i=0;
  591. /* Vertex index should already correct. */
  592. //for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ mv->i=i; i++; }
  593. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ tnsVert* v=tnsFillVert(mo, LA_COLOR3(mv->p)); tnsFillVertNormal(v,mv->n); v->flags=mv->flags; }
  594. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ tnsFace* f=tnsFillFace(mo, mf->looplen, -1); f->flags=mf->flags;
  595. int j=0; for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  596. laListItemPointer* next=lip->pNext; if(!next) next=mf->l.pFirst; tnsMEdge* me0=lip->p, *me1=next->p;
  597. tnsFillFaceLoop(f, j, tnsMMeshEdgeStartingVert(me0,me1)->i); j++;
  598. }
  599. tnsFillFaceNormal(f,mf->n);
  600. }
  601. mo->totv=mo->totmv; mo->totf=mo->totmf;
  602. mo->maxv=mo->totv; mo->maxe=mo->tote; mo->maxf=mo->totf;
  603. if((!mo->maxv) && mo->v) arrFree(&mo->v, &mo->maxv);
  604. if((!mo->maxe) && mo->e) arrFree(&mo->e, &mo->maxe);
  605. if((!mo->maxf) && mo->f) arrFree(&mo->f, &mo->maxf);
  606. tnsClearMMesh(mo);
  607. }
  608. void tnsMeshEnterEditMode(tnsMeshObject* mo){
  609. if(mo->Mode==TNS_MESH_EDIT_MODE) return;
  610. tnsMMeshFromMesh(mo);
  611. mo->Mode = TNS_MESH_EDIT_MODE;
  612. tnsInvalidateMeshBatch(mo);
  613. tnsInvalidateEvaluation(mo);
  614. }
  615. void tnsMeshLeaveEditMode(tnsMeshObject* mo){
  616. if(mo->Mode==TNS_MESH_OBJECT_MODE) return;
  617. tnsMeshFromMMesh(mo);
  618. mo->Mode = TNS_MESH_OBJECT_MODE;
  619. tnsInvalidateMeshBatch(mo);
  620. tnsInvalidateEvaluation(mo);
  621. }
  622. void tnsMMeshClearFirstLastSelection(tnsMeshObject* mo){
  623. mo->FirstSelectE=mo->FirstSelectV=mo->LastSelectE=mo->LastSelectV=0;
  624. }
  625. int tnsMMeshAnySelected(tnsMeshObject* mo){
  626. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ if(mv->flags&TNS_MESH_FLAG_SELECTED) return 1; }
  627. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ if(me->flags&TNS_MESH_FLAG_SELECTED) return 1; }
  628. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ if(mf->flags&TNS_MESH_FLAG_SELECTED) return 1; } return 0;
  629. }
  630. void tnsMMeshClearExtraFlags(tnsMeshObject* mo){
  631. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ mv->flags&=(~TNS_MESH_FLAG_PICKED); }
  632. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ me->flags&=(~TNS_MESH_FLAG_PICKED); /*me->flags&=(~TNS_MESH_FLAG_LOOP_REVERSE);*/ }
  633. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ mf->flags&=(~TNS_MESH_FLAG_PICKED); }
  634. }
  635. void tnsMMeshDeselectAll(tnsMeshObject* mo){
  636. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ mv->flags&=(~TNS_MESH_FLAG_SELECTED); }
  637. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ me->flags&=(~TNS_MESH_FLAG_SELECTED); }
  638. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ mf->flags&=(~TNS_MESH_FLAG_SELECTED); }
  639. tnsMMeshClearFirstLastSelection(mo);
  640. }
  641. void tnsMMeshSelectAll(tnsMeshObject* mo){
  642. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ mv->flags|=TNS_MESH_FLAG_SELECTED; }
  643. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ me->flags|=TNS_MESH_FLAG_SELECTED; }
  644. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ mf->flags|=TNS_MESH_FLAG_SELECTED; }
  645. }
  646. void tnsMMeshSelectVert(tnsMeshObject* mo, tnsMVert* mv, int select, int toggle){
  647. if(!mo) return;
  648. if(toggle) tnsMMeshSelectVert(mo,mv,(mv->flags&TNS_MESH_FLAG_SELECTED?0:1),0);
  649. elif(select) mv->flags|=TNS_MESH_FLAG_SELECTED; else mv->flags&=(~TNS_MESH_FLAG_SELECTED);
  650. if(!mo->FirstSelectV) mo->FirstSelectV=mv; mo->LastSelectV=mv;
  651. }
  652. void tnsMMeshSelectEdge(tnsMeshObject* mo, tnsMEdge* me, int select, int toggle){
  653. if(!mo) return;
  654. if(toggle) tnsMMeshSelectEdge(mo,me,(me->flags&TNS_MESH_FLAG_SELECTED?0:1),0);
  655. elif(select) me->flags|=TNS_MESH_FLAG_SELECTED; else me->flags&=(~TNS_MESH_FLAG_SELECTED);
  656. if(!mo->FirstSelectE) mo->FirstSelectE=me; mo->LastSelectE=me;
  657. }
  658. void tnsMMeshEnsureSelectionFromVerts(tnsMeshObject* mo){
  659. for(tnsMEdge* me=mo->me.pFirst;me;me=me->Item.pNext){ me->flags&=(~TNS_MESH_FLAG_SELECTED);
  660. if(me->vl->flags&me->vr->flags&TNS_MESH_FLAG_SELECTED) me->flags|=TNS_MESH_FLAG_SELECTED;
  661. }
  662. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ int sel=1; mf->flags&=(~TNS_MESH_FLAG_SELECTED);
  663. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){ tnsMEdge*me=lip->p; if(!(me->flags&TNS_MESH_FLAG_SELECTED)){ sel=0; break; } }
  664. if(sel){ mf->flags|=TNS_MESH_FLAG_SELECTED; }
  665. }
  666. }
  667. void tnsMMeshEnsureSelectionFromEdges(tnsMeshObject* mo){
  668. for(tnsMVert* mv=mo->mv.pFirst;mv;mv=mv->Item.pNext){ mv->flags&=(~TNS_MESH_FLAG_SELECTED);
  669. for(laListItemPointer* lip=mv->elink.pFirst;lip;lip=lip->pNext){ tnsMEdge*me=lip->p; if(me->flags&TNS_MESH_FLAG_SELECTED){ mv->flags|=TNS_MESH_FLAG_SELECTED; break; } }
  670. }
  671. for(tnsMFace* mf=mo->mf.pFirst;mf;mf=mf->Item.pNext){ int sel=1; mf->flags&=(~TNS_MESH_FLAG_SELECTED);
  672. for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){ tnsMEdge*me=lip->p; if(!(me->flags&TNS_MESH_FLAG_SELECTED)){ sel=0; break; } }
  673. if(sel){ mf->flags|=TNS_MESH_FLAG_SELECTED; }
  674. }
  675. }
  676. void tnsMMeshEnsureSelection(tnsMeshObject* mo, int SelectMode){
  677. if(SelectMode==LA_CANVAS_SELECT_MODE_VERTS){ tnsMMeshEnsureSelectionFromVerts(mo); }
  678. elif(SelectMode==LA_CANVAS_SELECT_MODE_EDGES){ tnsMMeshEnsureSelectionFromEdges(mo); }
  679. }
  680. tnsMEdge* tnsMMeshGetRingEdge(tnsMEdge* me, int from_right){
  681. tnsMEdge* candidate=0; tnsMVert* VL=from_right?me->vr:me->vl, *VR=from_right?me->vl:me->vr;
  682. for(laListItemPointer* lip=VL->elink.pFirst;lip;lip=lip->pNext){
  683. tnsMEdge* te=lip->p; if(te==me || (te->flags&TNS_MESH_FLAG_PICKED)){ continue; }
  684. if((!te->fl) || (!te->fr)){ candidate=te; break; }
  685. }
  686. if(candidate){ candidate->flags|=TNS_MESH_FLAG_PICKED; return candidate; }
  687. if(lstCountElements(&VL->elink)!=4) return 0;
  688. tnsMFace* f1=0,*f2=0;
  689. for(laListItemPointer* lip=VL->elink.pFirst;lip;lip=lip->pNext){
  690. tnsMEdge* te=lip->p; if(te==me || (te->flags&TNS_MESH_FLAG_PICKED)){ continue; }
  691. if(tnsMMeshFaceHasVert(te->fl,VR) || tnsMMeshFaceHasVert(te->fr,VR)){ continue; }
  692. else{ te->flags|=TNS_MESH_FLAG_PICKED; return te; }
  693. }
  694. return 0;
  695. }
  696. void tnsMMeshExpandRingList(tnsMeshObject* mo, tnsMEdge* me, laListHandle* lst){
  697. tnsMEdge* el,*er;
  698. int flc=0,frc=0; if(me->fl) flc=me->fl->looplen; if(me->fr) frc=me->fr->looplen;
  699. if(TNS_MAX2(flc,frc)>4){
  700. if(flc>frc){ for(laListItemPointer* lip=me->fl->l.pFirst;lip;lip=lip->pNext){ lstAppendPointer(lst,lip->p); } return; }
  701. elif(flc<frc){ for(laListItemPointer* lip=me->fr->l.pFirst;lip;lip=lip->pNext){ lstAppendPointer(lst,lip->p); } return; }
  702. }
  703. if(el=tnsMMeshGetRingEdge(me,0)){ lstAppendPointer(lst,el); tnsMMeshExpandRingList(mo,el,lst); }
  704. if(er=tnsMMeshGetRingEdge(me,1)){ lstAppendPointer(lst,er); tnsMMeshExpandRingList(mo,er,lst); }
  705. }
  706. tnsMEdge* tnsMMeshGetBandEdge(tnsMEdge* me, int from_right){
  707. tnsMFace* FL=from_right?me->fl:me->fr; if(!FL) return 0;
  708. if(FL->looplen!=4){ return 0;}
  709. for(laListItemPointer* lip=FL->l.pFirst;lip;lip=lip->pNext){
  710. tnsMEdge* te=lip->p; if((!(te->flags&TNS_MESH_FLAG_PICKED)) && (!tnsMMeshEdgeShareVert(te,me))){
  711. te->flags|=TNS_MESH_FLAG_PICKED; return te;
  712. }
  713. }
  714. return 0;
  715. }
  716. void tnsMMeshExpandBandList(tnsMeshObject* mo, tnsMEdge* me, laListHandle* lst){
  717. tnsMEdge* el,*er;
  718. if(el=tnsMMeshGetBandEdge(me,0)){ lstAppendPointer(lst,el); tnsMMeshExpandBandList(mo,el,lst); }
  719. if(er=tnsMMeshGetBandEdge(me,1)){ lstAppendPointer(lst,er); tnsMMeshExpandBandList(mo,er,lst); }
  720. }
  721. int tnsMMeshSelectRingBandFrom(tnsMeshObject* mo, tnsMEdge* me, int ring_band, int select, int toggle){
  722. if(!ring_band) return 0;
  723. tnsMMeshClearExtraFlags(mo);
  724. laListHandle lst={0}; lstAppendPointer(&lst,me); me->flags|=TNS_MESH_FLAG_PICKED;
  725. if(ring_band==1) tnsMMeshExpandRingList(mo,me,&lst);
  726. elif(ring_band==2) tnsMMeshExpandBandList(mo,me,&lst);
  727. if(lst.pFirst==lst.pLast){ while(lstPopPointer(&lst)); return 0; }
  728. int has_idle=0;
  729. if(toggle){
  730. for(laListItemPointer* lip=lst.pFirst;lip;lip=lip->pNext){
  731. tnsMEdge* te=lip->p; if(te!=me && (!(te->flags&TNS_MESH_FLAG_SELECTED))) has_idle=1;
  732. if(has_idle){ break; }
  733. }
  734. if(has_idle){ select=1; }else{ select=0; }
  735. }
  736. for(laListItemPointer* lip=lst.pFirst;lip;lip=lip->pNext){
  737. tnsMEdge* te=lip->p;
  738. if(select){ te->flags|=TNS_MESH_FLAG_SELECTED; te->vl->flags|=TNS_MESH_FLAG_SELECTED; te->vr->flags|=TNS_MESH_FLAG_SELECTED; }
  739. else{ te->flags&=(~TNS_MESH_FLAG_SELECTED); te->vl->flags&=(~TNS_MESH_FLAG_SELECTED); te->vr->flags&=(~TNS_MESH_FLAG_SELECTED); }
  740. }
  741. while(lstPopPointer(&lst)); return 1;
  742. }
  743. void tnsMMeshReduceFaceEdge(tnsMeshObject* mo, tnsMFace* mf, tnsMEdge* me){
  744. lstRemovePointerLeave(&mf->l,me); mf->looplen--;
  745. }
  746. void tnsMMeshFaceReplaceEdgeWith(tnsMFace* mf, tnsMEdge* to_be_replaced, tnsMEdge* as){
  747. if(!mf) return; for(laListItemPointer* lip=mf->l.pFirst;lip;lip=lip->pNext){
  748. if(lip->p==to_be_replaced){ lip->p=as; return; }
  749. }
  750. }
  751. void tnsMMeshReduceZippedFace(tnsMeshObject* mo, tnsMFace* mf){
  752. if(mf->looplen>2) return; if(mf->looplen<2){ printf("mf->looplen < 2 ?\n");return; }
  753. tnsMEdge* me1=((laListItemPointer*)mf->l.pFirst)->p,*me2=((laListItemPointer*)mf->l.pLast)->p;
  754. tnsMVert* vl=me1->vl; tnsMVert*vr=me1->vr;
  755. lstRemovePointerLeave(&vl->elink,me2); lstRemovePointerLeave(&vr->elink,me2);
  756. tnsMMeshFaceReplaceEdgeWith(me1->fl,me2,me1); tnsMMeshFaceReplaceEdgeWith(me1->fr,me2,me1);
  757. tnsMMeshFaceReplaceEdgeWith(me2->fl,me2,me1); tnsMMeshFaceReplaceEdgeWith(me2->fr,me2,me1);
  758. if(me1->fl==mf){ me1->fl=((me2->fl==mf)?me2->fr:me2->fl); }elif(me1->fr==mf){ me1->fr=((me2->fl==mf)?me2->fr:me2->fl); }
  759. lstRemoveItem(&mo->me,me2); mo->totme--; memLeave(me2);
  760. lstRemoveItem(&mo->mf,mf); mo->totmf--; memLeave(mf);
  761. tnsMMeshClearFirstLastSelection(mo);
  762. }
  763. int tnsMMeshVertsCanMerge(tnsMeshObject* mo, tnsMVert* into, tnsMVert* mv){
  764. for(laListItemPointer* lip=into->elink.pFirst;lip;lip=lip->pNext){ tnsMEdge* me=lip->p;
  765. for(laListItemPointer* lip2=mv->elink.pFirst;lip2;lip2=lip2->pNext){ tnsMEdge* me2=lip2->p; if(me==me2) continue;
  766. tnsMVert* mvs=tnsMMeshEdgeShareVert(me,me2); if((!mvs)||mvs==mv||mvs==into) continue; int count=0;
  767. if(me->fl) count++; if(me->fr) count++; if(me2->fl) count++; if(me2->fr) count++;
  768. tnsMFace* sf=tnsMMeshEdgeShareFace(me,me2); if(sf){count--;}
  769. if ((sf==me->fl && me->fr && (me->fr==me2->fl||me->fr==me2->fr))||
  770. (sf==me->fr && me->fl && (me->fl==me2->fl||me->fl==me2->fr))){ return 0; }
  771. if((sf && count>3) || (!sf && count>2)){ return 0; }
  772. }
  773. }
  774. return 1;
  775. }
  776. int tnsMMeshMergeVerts(tnsMeshObject* mo, tnsMVert* into, tnsMVert* mv){
  777. if(!tnsMMeshVertsCanMerge(mo,into,mv)) return 0;
  778. tnsMEdge* me=tnsMMeshMakeEdge(mo,into,mv);
  779. if(me->fl){ tnsMMeshReduceFaceEdge(mo,me->fl,me); }
  780. if(me->fr){ tnsMMeshReduceFaceEdge(mo,me->fr,me); }
  781. tnsMEdge* me2; while(me2=lstPopPointerLeave(&mv->elink)){
  782. if(me2==me) continue; lstAppendPointer(&into->elink,me2);
  783. if(me2->vl==mv){ me2->vl=into; }elif(me2->vr==mv){ me2->vr=into; }
  784. }
  785. if(me->fl){ tnsMMeshReduceZippedFace(mo,me->fl); }
  786. if(me->fr){ tnsMMeshReduceZippedFace(mo,me->fr); }
  787. lstRemovePointerLeave(&into->elink,me);
  788. lstRemoveItem(&mo->mv,mv); memLeave(mv); mo->totmv--;
  789. lstRemoveItem(&mo->me,me); memLeave(me); mo->totme--;
  790. tnsMMeshClearFirstLastSelection(mo);
  791. //tnsPrintMeshTopology(mo);
  792. return 1;
  793. }
  794. tnsMeshObject *tnsCreateMeshEmpty(tnsObject *under, char *Name, real AtX, real AtY, real AtZ){
  795. tnsMeshObject *mo = memAcquireHyper(sizeof(tnsMeshObject));
  796. tnsInitObjectBase(&mo->Base, under, Name, TNS_OBJECT_MESH, AtX, AtY, AtZ, 0, 0, 0, 1.0f, TNS_ROTATION_XYZ_EULER, 1.0f);
  797. return mo;
  798. }
  799. tnsMeshObject *tnsCreateMeshPlane(tnsObject *under, char *Name, real AtX, real AtY, real AtZ, real size){
  800. tnsMeshObject *mo=tnsCreateMeshEmpty(under, Name, AtX, AtY, AtZ);
  801. tnsInitMeshPlane(mo, size);
  802. tnsInvalidateMeshBatch(mo);
  803. return mo;
  804. }