Mesh.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500
  1. /*
  2. ** SGI FREE SOFTWARE LICENSE B (Version 2.0, Sept. 18, 2008)
  3. ** Copyright (C) 2011 Silicon Graphics, Inc.
  4. ** All Rights Reserved.
  5. **
  6. ** Permission is hereby granted, free of charge, to any person obtaining a copy
  7. ** of this software and associated documentation files (the "Software"), to deal
  8. ** in the Software without restriction, including without limitation the rights
  9. ** to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
  10. ** of the Software, and to permit persons to whom the Software is furnished to do so,
  11. ** subject to the following conditions:
  12. **
  13. ** The above copyright notice including the dates of first publication and either this
  14. ** permission notice or a reference to http://oss.sgi.com/projects/FreeB/ shall be
  15. ** included in all copies or substantial portions of the Software.
  16. **
  17. ** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
  18. ** INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
  19. ** PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL SILICON GRAPHICS, INC.
  20. ** BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
  21. ** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
  22. ** OR OTHER DEALINGS IN THE SOFTWARE.
  23. **
  24. ** Except as contained in this notice, the name of Silicon Graphics, Inc. shall not
  25. ** be used in advertising or otherwise to promote the sale, use or other dealings in
  26. ** this Software without prior written authorization from Silicon Graphics, Inc.
  27. */
  28. /*
  29. ** Original Author: Eric Veach, July 1994.
  30. ** libtess2: Mikko Mononen, http://code.google.com/p/libtess2/.
  31. ** LibTessDotNet: Remi Gillig, https://github.com/speps/LibTessDotNet
  32. */
  33. using System;
  34. using System.Diagnostics;
  35. namespace UnityEngine.Experimental.Rendering.Universal
  36. {
  37. namespace LibTessDotNet
  38. {
  39. internal class Mesh : MeshUtils.Pooled<Mesh>
  40. {
  41. internal MeshUtils.Vertex _vHead;
  42. internal MeshUtils.Face _fHead;
  43. internal MeshUtils.Edge _eHead, _eHeadSym;
  44. public Mesh()
  45. {
  46. var v = _vHead = MeshUtils.Vertex.Create();
  47. var f = _fHead = MeshUtils.Face.Create();
  48. var pair = MeshUtils.EdgePair.Create();
  49. var e = _eHead = pair._e;
  50. var eSym = _eHeadSym = pair._eSym;
  51. v._next = v._prev = v;
  52. v._anEdge = null;
  53. f._next = f._prev = f;
  54. f._anEdge = null;
  55. f._trail = null;
  56. f._marked = false;
  57. f._inside = false;
  58. e._next = e;
  59. e._Sym = eSym;
  60. e._Onext = null;
  61. e._Lnext = null;
  62. e._Org = null;
  63. e._Lface = null;
  64. e._winding = 0;
  65. e._activeRegion = null;
  66. eSym._next = eSym;
  67. eSym._Sym = e;
  68. eSym._Onext = null;
  69. eSym._Lnext = null;
  70. eSym._Org = null;
  71. eSym._Lface = null;
  72. eSym._winding = 0;
  73. eSym._activeRegion = null;
  74. }
  75. public override void Reset()
  76. {
  77. _vHead = null;
  78. _fHead = null;
  79. _eHead = _eHeadSym = null;
  80. }
  81. public override void OnFree()
  82. {
  83. for (MeshUtils.Face f = _fHead._next, fNext = _fHead; f != _fHead; f = fNext)
  84. {
  85. fNext = f._next;
  86. f.Free();
  87. }
  88. for (MeshUtils.Vertex v = _vHead._next, vNext = _vHead; v != _vHead; v = vNext)
  89. {
  90. vNext = v._next;
  91. v.Free();
  92. }
  93. for (MeshUtils.Edge e = _eHead._next, eNext = _eHead; e != _eHead; e = eNext)
  94. {
  95. eNext = e._next;
  96. e.Free();
  97. }
  98. }
  99. /// <summary>
  100. /// Creates one edge, two vertices and a loop (face).
  101. /// The loop consists of the two new half-edges.
  102. /// </summary>
  103. public MeshUtils.Edge MakeEdge()
  104. {
  105. var e = MeshUtils.MakeEdge(_eHead);
  106. MeshUtils.MakeVertex(e, _vHead);
  107. MeshUtils.MakeVertex(e._Sym, _vHead);
  108. MeshUtils.MakeFace(e, _fHead);
  109. return e;
  110. }
  111. /// <summary>
  112. /// Splice is the basic operation for changing the
  113. /// mesh connectivity and topology. It changes the mesh so that
  114. /// eOrg->Onext = OLD( eDst->Onext )
  115. /// eDst->Onext = OLD( eOrg->Onext )
  116. /// where OLD(...) means the value before the meshSplice operation.
  117. ///
  118. /// This can have two effects on the vertex structure:
  119. /// - if eOrg->Org != eDst->Org, the two vertices are merged together
  120. /// - if eOrg->Org == eDst->Org, the origin is split into two vertices
  121. /// In both cases, eDst->Org is changed and eOrg->Org is untouched.
  122. ///
  123. /// Similarly (and independently) for the face structure,
  124. /// - if eOrg->Lface == eDst->Lface, one loop is split into two
  125. /// - if eOrg->Lface != eDst->Lface, two distinct loops are joined into one
  126. /// In both cases, eDst->Lface is changed and eOrg->Lface is unaffected.
  127. ///
  128. /// Some special cases:
  129. /// If eDst == eOrg, the operation has no effect.
  130. /// If eDst == eOrg->Lnext, the new face will have a single edge.
  131. /// If eDst == eOrg->Lprev, the old face will have a single edge.
  132. /// If eDst == eOrg->Onext, the new vertex will have a single edge.
  133. /// If eDst == eOrg->Oprev, the old vertex will have a single edge.
  134. /// </summary>
  135. public void Splice(MeshUtils.Edge eOrg, MeshUtils.Edge eDst)
  136. {
  137. if (eOrg == eDst)
  138. {
  139. return;
  140. }
  141. bool joiningVertices = false;
  142. if (eDst._Org != eOrg._Org)
  143. {
  144. // We are merging two disjoint vertices -- destroy eDst->Org
  145. joiningVertices = true;
  146. MeshUtils.KillVertex(eDst._Org, eOrg._Org);
  147. }
  148. bool joiningLoops = false;
  149. if (eDst._Lface != eOrg._Lface)
  150. {
  151. // We are connecting two disjoint loops -- destroy eDst->Lface
  152. joiningLoops = true;
  153. MeshUtils.KillFace(eDst._Lface, eOrg._Lface);
  154. }
  155. // Change the edge structure
  156. MeshUtils.Splice(eDst, eOrg);
  157. if (!joiningVertices)
  158. {
  159. // We split one vertex into two -- the new vertex is eDst->Org.
  160. // Make sure the old vertex points to a valid half-edge.
  161. MeshUtils.MakeVertex(eDst, eOrg._Org);
  162. eOrg._Org._anEdge = eOrg;
  163. }
  164. if (!joiningLoops)
  165. {
  166. // We split one loop into two -- the new loop is eDst->Lface.
  167. // Make sure the old face points to a valid half-edge.
  168. MeshUtils.MakeFace(eDst, eOrg._Lface);
  169. eOrg._Lface._anEdge = eOrg;
  170. }
  171. }
  172. /// <summary>
  173. /// Removes the edge eDel. There are several cases:
  174. /// if (eDel->Lface != eDel->Rface), we join two loops into one; the loop
  175. /// eDel->Lface is deleted. Otherwise, we are splitting one loop into two;
  176. /// the newly created loop will contain eDel->Dst. If the deletion of eDel
  177. /// would create isolated vertices, those are deleted as well.
  178. /// </summary>
  179. public void Delete(MeshUtils.Edge eDel)
  180. {
  181. var eDelSym = eDel._Sym;
  182. // First step: disconnect the origin vertex eDel->Org. We make all
  183. // changes to get a consistent mesh in this "intermediate" state.
  184. bool joiningLoops = false;
  185. if (eDel._Lface != eDel._Rface)
  186. {
  187. // We are joining two loops into one -- remove the left face
  188. joiningLoops = true;
  189. MeshUtils.KillFace(eDel._Lface, eDel._Rface);
  190. }
  191. if (eDel._Onext == eDel)
  192. {
  193. MeshUtils.KillVertex(eDel._Org, null);
  194. }
  195. else
  196. {
  197. // Make sure that eDel->Org and eDel->Rface point to valid half-edges
  198. eDel._Rface._anEdge = eDel._Oprev;
  199. eDel._Org._anEdge = eDel._Onext;
  200. MeshUtils.Splice(eDel, eDel._Oprev);
  201. if (!joiningLoops)
  202. {
  203. // We are splitting one loop into two -- create a new loop for eDel.
  204. MeshUtils.MakeFace(eDel, eDel._Lface);
  205. }
  206. }
  207. // Claim: the mesh is now in a consistent state, except that eDel->Org
  208. // may have been deleted. Now we disconnect eDel->Dst.
  209. if (eDelSym._Onext == eDelSym)
  210. {
  211. MeshUtils.KillVertex(eDelSym._Org, null);
  212. MeshUtils.KillFace(eDelSym._Lface, null);
  213. }
  214. else
  215. {
  216. // Make sure that eDel->Dst and eDel->Lface point to valid half-edges
  217. eDel._Lface._anEdge = eDelSym._Oprev;
  218. eDelSym._Org._anEdge = eDelSym._Onext;
  219. MeshUtils.Splice(eDelSym, eDelSym._Oprev);
  220. }
  221. // Any isolated vertices or faces have already been freed.
  222. MeshUtils.KillEdge(eDel);
  223. }
  224. /// <summary>
  225. /// Creates a new edge such that eNew == eOrg.Lnext and eNew.Dst is a newly created vertex.
  226. /// eOrg and eNew will have the same left face.
  227. /// </summary>
  228. public MeshUtils.Edge AddEdgeVertex(MeshUtils.Edge eOrg)
  229. {
  230. var eNew = MeshUtils.MakeEdge(eOrg);
  231. var eNewSym = eNew._Sym;
  232. // Connect the new edge appropriately
  233. MeshUtils.Splice(eNew, eOrg._Lnext);
  234. // Set vertex and face information
  235. eNew._Org = eOrg._Dst;
  236. MeshUtils.MakeVertex(eNewSym, eNew._Org);
  237. eNew._Lface = eNewSym._Lface = eOrg._Lface;
  238. return eNew;
  239. }
  240. /// <summary>
  241. /// Splits eOrg into two edges eOrg and eNew such that eNew == eOrg.Lnext.
  242. /// The new vertex is eOrg.Dst == eNew.Org.
  243. /// eOrg and eNew will have the same left face.
  244. /// </summary>
  245. public MeshUtils.Edge SplitEdge(MeshUtils.Edge eOrg)
  246. {
  247. var eTmp = AddEdgeVertex(eOrg);
  248. var eNew = eTmp._Sym;
  249. // Disconnect eOrg from eOrg->Dst and connect it to eNew->Org
  250. MeshUtils.Splice(eOrg._Sym, eOrg._Sym._Oprev);
  251. MeshUtils.Splice(eOrg._Sym, eNew);
  252. // Set the vertex and face information
  253. eOrg._Dst = eNew._Org;
  254. eNew._Dst._anEdge = eNew._Sym; // may have pointed to eOrg->Sym
  255. eNew._Rface = eOrg._Rface;
  256. eNew._winding = eOrg._winding; // copy old winding information
  257. eNew._Sym._winding = eOrg._Sym._winding;
  258. return eNew;
  259. }
  260. /// <summary>
  261. /// Creates a new edge from eOrg->Dst to eDst->Org, and returns the corresponding half-edge eNew.
  262. /// If eOrg->Lface == eDst->Lface, this splits one loop into two,
  263. /// and the newly created loop is eNew->Lface. Otherwise, two disjoint
  264. /// loops are merged into one, and the loop eDst->Lface is destroyed.
  265. ///
  266. /// If (eOrg == eDst), the new face will have only two edges.
  267. /// If (eOrg->Lnext == eDst), the old face is reduced to a single edge.
  268. /// If (eOrg->Lnext->Lnext == eDst), the old face is reduced to two edges.
  269. /// </summary>
  270. public MeshUtils.Edge Connect(MeshUtils.Edge eOrg, MeshUtils.Edge eDst)
  271. {
  272. var eNew = MeshUtils.MakeEdge(eOrg);
  273. var eNewSym = eNew._Sym;
  274. bool joiningLoops = false;
  275. if (eDst._Lface != eOrg._Lface)
  276. {
  277. // We are connecting two disjoint loops -- destroy eDst->Lface
  278. joiningLoops = true;
  279. MeshUtils.KillFace(eDst._Lface, eOrg._Lface);
  280. }
  281. // Connect the new edge appropriately
  282. MeshUtils.Splice(eNew, eOrg._Lnext);
  283. MeshUtils.Splice(eNewSym, eDst);
  284. // Set the vertex and face information
  285. eNew._Org = eOrg._Dst;
  286. eNewSym._Org = eDst._Org;
  287. eNew._Lface = eNewSym._Lface = eOrg._Lface;
  288. // Make sure the old face points to a valid half-edge
  289. eOrg._Lface._anEdge = eNewSym;
  290. if (!joiningLoops)
  291. {
  292. MeshUtils.MakeFace(eNew, eOrg._Lface);
  293. }
  294. return eNew;
  295. }
  296. /// <summary>
  297. /// Destroys a face and removes it from the global face list. All edges of
  298. /// fZap will have a NULL pointer as their left face. Any edges which
  299. /// also have a NULL pointer as their right face are deleted entirely
  300. /// (along with any isolated vertices this produces).
  301. /// An entire mesh can be deleted by zapping its faces, one at a time,
  302. /// in any order. Zapped faces cannot be used in further mesh operations!
  303. /// </summary>
  304. public void ZapFace(MeshUtils.Face fZap)
  305. {
  306. var eStart = fZap._anEdge;
  307. // walk around face, deleting edges whose right face is also NULL
  308. var eNext = eStart._Lnext;
  309. MeshUtils.Edge e, eSym;
  310. do {
  311. e = eNext;
  312. eNext = e._Lnext;
  313. e._Lface = null;
  314. if (e._Rface == null)
  315. {
  316. // delete the edge -- see TESSmeshDelete above
  317. if (e._Onext == e)
  318. {
  319. MeshUtils.KillVertex(e._Org, null);
  320. }
  321. else
  322. {
  323. // Make sure that e._Org points to a valid half-edge
  324. e._Org._anEdge = e._Onext;
  325. MeshUtils.Splice(e, e._Oprev);
  326. }
  327. eSym = e._Sym;
  328. if (eSym._Onext == eSym)
  329. {
  330. MeshUtils.KillVertex(eSym._Org, null);
  331. }
  332. else
  333. {
  334. // Make sure that eSym._Org points to a valid half-edge
  335. eSym._Org._anEdge = eSym._Onext;
  336. MeshUtils.Splice(eSym, eSym._Oprev);
  337. }
  338. MeshUtils.KillEdge(e);
  339. }
  340. } while (e != eStart);
  341. /* delete from circular doubly-linked list */
  342. var fPrev = fZap._prev;
  343. var fNext = fZap._next;
  344. fNext._prev = fPrev;
  345. fPrev._next = fNext;
  346. fZap.Free();
  347. }
  348. public void MergeConvexFaces(int maxVertsPerFace)
  349. {
  350. for (var f = _fHead._next; f != _fHead; f = f._next)
  351. {
  352. // Skip faces which are outside the result
  353. if (!f._inside)
  354. {
  355. continue;
  356. }
  357. var eCur = f._anEdge;
  358. var vStart = eCur._Org;
  359. while (true)
  360. {
  361. var eNext = eCur._Lnext;
  362. var eSym = eCur._Sym;
  363. if (eSym != null && eSym._Lface != null && eSym._Lface._inside)
  364. {
  365. // Try to merge the neighbour faces if the resulting polygons
  366. // does not exceed maximum number of vertices.
  367. int curNv = f.VertsCount;
  368. int symNv = eSym._Lface.VertsCount;
  369. if ((curNv + symNv - 2) <= maxVertsPerFace)
  370. {
  371. // Merge if the resulting poly is convex.
  372. if (Geom.VertCCW(eCur._Lprev._Org, eCur._Org, eSym._Lnext._Lnext._Org) &&
  373. Geom.VertCCW(eSym._Lprev._Org, eSym._Org, eCur._Lnext._Lnext._Org))
  374. {
  375. eNext = eSym._Lnext;
  376. Delete(eSym);
  377. eCur = null;
  378. }
  379. }
  380. }
  381. if (eCur != null && eCur._Lnext._Org == vStart)
  382. break;
  383. // Continue to next edge.
  384. eCur = eNext;
  385. }
  386. }
  387. }
  388. [Conditional("DEBUG")]
  389. public void Check()
  390. {
  391. MeshUtils.Edge e;
  392. MeshUtils.Face fPrev = _fHead, f;
  393. for (fPrev = _fHead; (f = fPrev._next) != _fHead; fPrev = f)
  394. {
  395. e = f._anEdge;
  396. do {
  397. Debug.Assert(e._Sym != e);
  398. Debug.Assert(e._Sym._Sym == e);
  399. Debug.Assert(e._Lnext._Onext._Sym == e);
  400. Debug.Assert(e._Onext._Sym._Lnext == e);
  401. Debug.Assert(e._Lface == f);
  402. e = e._Lnext;
  403. } while (e != f._anEdge);
  404. }
  405. Debug.Assert(f._prev == fPrev && f._anEdge == null);
  406. MeshUtils.Vertex vPrev = _vHead, v;
  407. for (vPrev = _vHead; (v = vPrev._next) != _vHead; vPrev = v)
  408. {
  409. Debug.Assert(v._prev == vPrev);
  410. e = v._anEdge;
  411. do
  412. {
  413. Debug.Assert(e._Sym != e);
  414. Debug.Assert(e._Sym._Sym == e);
  415. Debug.Assert(e._Lnext._Onext._Sym == e);
  416. Debug.Assert(e._Onext._Sym._Lnext == e);
  417. Debug.Assert(e._Org == v);
  418. e = e._Onext;
  419. } while (e != v._anEdge);
  420. }
  421. Debug.Assert(v._prev == vPrev && v._anEdge == null);
  422. MeshUtils.Edge ePrev = _eHead;
  423. for (ePrev = _eHead; (e = ePrev._next) != _eHead; ePrev = e)
  424. {
  425. Debug.Assert(e._Sym._next == ePrev._Sym);
  426. Debug.Assert(e._Sym != e);
  427. Debug.Assert(e._Sym._Sym == e);
  428. Debug.Assert(e._Org != null);
  429. Debug.Assert(e._Dst != null);
  430. Debug.Assert(e._Lnext._Onext._Sym == e);
  431. Debug.Assert(e._Onext._Sym._Lnext == e);
  432. }
  433. Debug.Assert(e._Sym._next == ePrev._Sym
  434. && e._Sym == _eHeadSym
  435. && e._Sym._Sym == e
  436. && e._Org == null && e._Dst == null
  437. && e._Lface == null && e._Rface == null);
  438. }
  439. }
  440. }
  441. }