Sweep.cs 51 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238
  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. using Real = System.Single;
  38. namespace LibTessDotNet
  39. {
  40. internal partial class Tess
  41. {
  42. internal class ActiveRegion
  43. {
  44. internal MeshUtils.Edge _eUp;
  45. internal Dict<ActiveRegion>.Node _nodeUp;
  46. internal int _windingNumber;
  47. internal bool _inside, _sentinel, _dirty, _fixUpperEdge;
  48. }
  49. private ActiveRegion RegionBelow(ActiveRegion reg)
  50. {
  51. return reg._nodeUp._prev._key;
  52. }
  53. private ActiveRegion RegionAbove(ActiveRegion reg)
  54. {
  55. return reg._nodeUp._next._key;
  56. }
  57. /// <summary>
  58. /// Both edges must be directed from right to left (this is the canonical
  59. /// direction for the upper edge of each region).
  60. ///
  61. /// The strategy is to evaluate a "t" value for each edge at the
  62. /// current sweep line position, given by tess->event. The calculations
  63. /// are designed to be very stable, but of course they are not perfect.
  64. ///
  65. /// Special case: if both edge destinations are at the sweep event,
  66. /// we sort the edges by slope (they would otherwise compare equally).
  67. /// </summary>
  68. private bool EdgeLeq(ActiveRegion reg1, ActiveRegion reg2)
  69. {
  70. var e1 = reg1._eUp;
  71. var e2 = reg2._eUp;
  72. if (e1._Dst == _event)
  73. {
  74. if (e2._Dst == _event)
  75. {
  76. // Two edges right of the sweep line which meet at the sweep event.
  77. // Sort them by slope.
  78. if (Geom.VertLeq(e1._Org, e2._Org))
  79. {
  80. return Geom.EdgeSign(e2._Dst, e1._Org, e2._Org) <= 0.0f;
  81. }
  82. return Geom.EdgeSign(e1._Dst, e2._Org, e1._Org) >= 0.0f;
  83. }
  84. return Geom.EdgeSign(e2._Dst, _event, e2._Org) <= 0.0f;
  85. }
  86. if (e2._Dst == _event)
  87. {
  88. return Geom.EdgeSign(e1._Dst, _event, e1._Org) >= 0.0f;
  89. }
  90. // General case - compute signed distance *from* e1, e2 to event
  91. var t1 = Geom.EdgeEval(e1._Dst, _event, e1._Org);
  92. var t2 = Geom.EdgeEval(e2._Dst, _event, e2._Org);
  93. return (t1 >= t2);
  94. }
  95. private void DeleteRegion(ActiveRegion reg)
  96. {
  97. if (reg._fixUpperEdge)
  98. {
  99. // It was created with zero winding number, so it better be
  100. // deleted with zero winding number (ie. it better not get merged
  101. // with a real edge).
  102. Debug.Assert(reg._eUp._winding == 0);
  103. }
  104. reg._eUp._activeRegion = null;
  105. _dict.Remove(reg._nodeUp);
  106. }
  107. /// <summary>
  108. /// Replace an upper edge which needs fixing (see ConnectRightVertex).
  109. /// </summary>
  110. private void FixUpperEdge(ActiveRegion reg, MeshUtils.Edge newEdge)
  111. {
  112. Debug.Assert(reg._fixUpperEdge);
  113. _mesh.Delete(reg._eUp);
  114. reg._fixUpperEdge = false;
  115. reg._eUp = newEdge;
  116. newEdge._activeRegion = reg;
  117. }
  118. private ActiveRegion TopLeftRegion(ActiveRegion reg)
  119. {
  120. var org = reg._eUp._Org;
  121. // Find the region above the uppermost edge with the same origin
  122. do {
  123. reg = RegionAbove(reg);
  124. } while (reg._eUp._Org == org);
  125. // If the edge above was a temporary edge introduced by ConnectRightVertex,
  126. // now is the time to fix it.
  127. if (reg._fixUpperEdge)
  128. {
  129. var e = _mesh.Connect(RegionBelow(reg)._eUp._Sym, reg._eUp._Lnext);
  130. FixUpperEdge(reg, e);
  131. reg = RegionAbove(reg);
  132. }
  133. return reg;
  134. }
  135. private ActiveRegion TopRightRegion(ActiveRegion reg)
  136. {
  137. var dst = reg._eUp._Dst;
  138. // Find the region above the uppermost edge with the same destination
  139. do {
  140. reg = RegionAbove(reg);
  141. } while (reg._eUp._Dst == dst);
  142. return reg;
  143. }
  144. /// <summary>
  145. /// Add a new active region to the sweep line, *somewhere* below "regAbove"
  146. /// (according to where the new edge belongs in the sweep-line dictionary).
  147. /// The upper edge of the new region will be "eNewUp".
  148. /// Winding number and "inside" flag are not updated.
  149. /// </summary>
  150. private ActiveRegion AddRegionBelow(ActiveRegion regAbove, MeshUtils.Edge eNewUp)
  151. {
  152. var regNew = new ActiveRegion();
  153. regNew._eUp = eNewUp;
  154. regNew._nodeUp = _dict.InsertBefore(regAbove._nodeUp, regNew);
  155. regNew._fixUpperEdge = false;
  156. regNew._sentinel = false;
  157. regNew._dirty = false;
  158. eNewUp._activeRegion = regNew;
  159. return regNew;
  160. }
  161. private void ComputeWinding(ActiveRegion reg)
  162. {
  163. reg._windingNumber = RegionAbove(reg)._windingNumber + reg._eUp._winding;
  164. reg._inside = Geom.IsWindingInside(_windingRule, reg._windingNumber);
  165. }
  166. /// <summary>
  167. /// Delete a region from the sweep line. This happens when the upper
  168. /// and lower chains of a region meet (at a vertex on the sweep line).
  169. /// The "inside" flag is copied to the appropriate mesh face (we could
  170. /// not do this before -- since the structure of the mesh is always
  171. /// changing, this face may not have even existed until now).
  172. /// </summary>
  173. private void FinishRegion(ActiveRegion reg)
  174. {
  175. var e = reg._eUp;
  176. var f = e._Lface;
  177. f._inside = reg._inside;
  178. f._anEdge = e;
  179. DeleteRegion(reg);
  180. }
  181. /// <summary>
  182. /// We are given a vertex with one or more left-going edges. All affected
  183. /// edges should be in the edge dictionary. Starting at regFirst->eUp,
  184. /// we walk down deleting all regions where both edges have the same
  185. /// origin vOrg. At the same time we copy the "inside" flag from the
  186. /// active region to the face, since at this point each face will belong
  187. /// to at most one region (this was not necessarily true until this point
  188. /// in the sweep). The walk stops at the region above regLast; if regLast
  189. /// is null we walk as far as possible. At the same time we relink the
  190. /// mesh if necessary, so that the ordering of edges around vOrg is the
  191. /// same as in the dictionary.
  192. /// </summary>
  193. private MeshUtils.Edge FinishLeftRegions(ActiveRegion regFirst, ActiveRegion regLast)
  194. {
  195. var regPrev = regFirst;
  196. var ePrev = regFirst._eUp;
  197. while (regPrev != regLast)
  198. {
  199. regPrev._fixUpperEdge = false; // placement was OK
  200. var reg = RegionBelow(regPrev);
  201. var e = reg._eUp;
  202. if (e._Org != ePrev._Org)
  203. {
  204. if (!reg._fixUpperEdge)
  205. {
  206. // Remove the last left-going edge. Even though there are no further
  207. // edges in the dictionary with this origin, there may be further
  208. // such edges in the mesh (if we are adding left edges to a vertex
  209. // that has already been processed). Thus it is important to call
  210. // FinishRegion rather than just DeleteRegion.
  211. FinishRegion(regPrev);
  212. break;
  213. }
  214. // If the edge below was a temporary edge introduced by
  215. // ConnectRightVertex, now is the time to fix it.
  216. e = _mesh.Connect(ePrev._Lprev, e._Sym);
  217. FixUpperEdge(reg, e);
  218. }
  219. // Relink edges so that ePrev.Onext == e
  220. if (ePrev._Onext != e)
  221. {
  222. _mesh.Splice(e._Oprev, e);
  223. _mesh.Splice(ePrev, e);
  224. }
  225. FinishRegion(regPrev); // may change reg.eUp
  226. ePrev = reg._eUp;
  227. regPrev = reg;
  228. }
  229. return ePrev;
  230. }
  231. /// <summary>
  232. /// Purpose: insert right-going edges into the edge dictionary, and update
  233. /// winding numbers and mesh connectivity appropriately. All right-going
  234. /// edges share a common origin vOrg. Edges are inserted CCW starting at
  235. /// eFirst; the last edge inserted is eLast.Oprev. If vOrg has any
  236. /// left-going edges already processed, then eTopLeft must be the edge
  237. /// such that an imaginary upward vertical segment from vOrg would be
  238. /// contained between eTopLeft.Oprev and eTopLeft; otherwise eTopLeft
  239. /// should be null.
  240. /// </summary>
  241. private void AddRightEdges(ActiveRegion regUp, MeshUtils.Edge eFirst, MeshUtils.Edge eLast, MeshUtils.Edge eTopLeft, bool cleanUp)
  242. {
  243. bool firstTime = true;
  244. var e = eFirst; do
  245. {
  246. Debug.Assert(Geom.VertLeq(e._Org, e._Dst));
  247. AddRegionBelow(regUp, e._Sym);
  248. e = e._Onext;
  249. } while (e != eLast);
  250. // Walk *all* right-going edges from e.Org, in the dictionary order,
  251. // updating the winding numbers of each region, and re-linking the mesh
  252. // edges to match the dictionary ordering (if necessary).
  253. if (eTopLeft == null)
  254. {
  255. eTopLeft = RegionBelow(regUp)._eUp._Rprev;
  256. }
  257. ActiveRegion regPrev = regUp, reg;
  258. var ePrev = eTopLeft;
  259. while (true)
  260. {
  261. reg = RegionBelow(regPrev);
  262. e = reg._eUp._Sym;
  263. if (e._Org != ePrev._Org) break;
  264. if (e._Onext != ePrev)
  265. {
  266. // Unlink e from its current position, and relink below ePrev
  267. _mesh.Splice(e._Oprev, e);
  268. _mesh.Splice(ePrev._Oprev, e);
  269. }
  270. // Compute the winding number and "inside" flag for the new regions
  271. reg._windingNumber = regPrev._windingNumber - e._winding;
  272. reg._inside = Geom.IsWindingInside(_windingRule, reg._windingNumber);
  273. // Check for two outgoing edges with same slope -- process these
  274. // before any intersection tests (see example in tessComputeInterior).
  275. regPrev._dirty = true;
  276. if (!firstTime && CheckForRightSplice(regPrev))
  277. {
  278. Geom.AddWinding(e, ePrev);
  279. DeleteRegion(regPrev);
  280. _mesh.Delete(ePrev);
  281. }
  282. firstTime = false;
  283. regPrev = reg;
  284. ePrev = e;
  285. }
  286. regPrev._dirty = true;
  287. Debug.Assert(regPrev._windingNumber - e._winding == reg._windingNumber);
  288. if (cleanUp)
  289. {
  290. // Check for intersections between newly adjacent edges.
  291. WalkDirtyRegions(regPrev);
  292. }
  293. }
  294. /// <summary>
  295. /// Two vertices with idential coordinates are combined into one.
  296. /// e1.Org is kept, while e2.Org is discarded.
  297. /// </summary>
  298. private void SpliceMergeVertices(MeshUtils.Edge e1, MeshUtils.Edge e2)
  299. {
  300. _mesh.Splice(e1, e2);
  301. }
  302. /// <summary>
  303. /// Find some weights which describe how the intersection vertex is
  304. /// a linear combination of "org" and "dest". Each of the two edges
  305. /// which generated "isect" is allocated 50% of the weight; each edge
  306. /// splits the weight between its org and dst according to the
  307. /// relative distance to "isect".
  308. /// </summary>
  309. private void VertexWeights(MeshUtils.Vertex isect, MeshUtils.Vertex org, MeshUtils.Vertex dst, out Real w0, out Real w1)
  310. {
  311. var t1 = Geom.VertL1dist(org, isect);
  312. var t2 = Geom.VertL1dist(dst, isect);
  313. w0 = (t2 / (t1 + t2)) / 2.0f;
  314. w1 = (t1 / (t1 + t2)) / 2.0f;
  315. isect._coords.X += w0 * org._coords.X + w1 * dst._coords.X;
  316. isect._coords.Y += w0 * org._coords.Y + w1 * dst._coords.Y;
  317. isect._coords.Z += w0 * org._coords.Z + w1 * dst._coords.Z;
  318. }
  319. /// <summary>
  320. /// We've computed a new intersection point, now we need a "data" pointer
  321. /// from the user so that we can refer to this new vertex in the
  322. /// rendering callbacks.
  323. /// </summary>
  324. private void GetIntersectData(MeshUtils.Vertex isect, MeshUtils.Vertex orgUp, MeshUtils.Vertex dstUp, MeshUtils.Vertex orgLo, MeshUtils.Vertex dstLo)
  325. {
  326. isect._coords = Vec3.Zero;
  327. Real w0, w1, w2, w3;
  328. VertexWeights(isect, orgUp, dstUp, out w0, out w1);
  329. VertexWeights(isect, orgLo, dstLo, out w2, out w3);
  330. if (_combineCallback != null)
  331. {
  332. isect._data = _combineCallback(
  333. isect._coords,
  334. new object[] { orgUp._data, dstUp._data, orgLo._data, dstLo._data },
  335. new Real[] { w0, w1, w2, w3 }
  336. );
  337. }
  338. }
  339. /// <summary>
  340. /// Check the upper and lower edge of "regUp", to make sure that the
  341. /// eUp->Org is above eLo, or eLo->Org is below eUp (depending on which
  342. /// origin is leftmost).
  343. ///
  344. /// The main purpose is to splice right-going edges with the same
  345. /// dest vertex and nearly identical slopes (ie. we can't distinguish
  346. /// the slopes numerically). However the splicing can also help us
  347. /// to recover from numerical errors. For example, suppose at one
  348. /// point we checked eUp and eLo, and decided that eUp->Org is barely
  349. /// above eLo. Then later, we split eLo into two edges (eg. from
  350. /// a splice operation like this one). This can change the result of
  351. /// our test so that now eUp->Org is incident to eLo, or barely below it.
  352. /// We must correct this condition to maintain the dictionary invariants.
  353. ///
  354. /// One possibility is to check these edges for intersection again
  355. /// (ie. CheckForIntersect). This is what we do if possible. However
  356. /// CheckForIntersect requires that tess->event lies between eUp and eLo,
  357. /// so that it has something to fall back on when the intersection
  358. /// calculation gives us an unusable answer. So, for those cases where
  359. /// we can't check for intersection, this routine fixes the problem
  360. /// by just splicing the offending vertex into the other edge.
  361. /// This is a guaranteed solution, no matter how degenerate things get.
  362. /// Basically this is a combinatorial solution to a numerical problem.
  363. /// </summary>
  364. private bool CheckForRightSplice(ActiveRegion regUp)
  365. {
  366. var regLo = RegionBelow(regUp);
  367. var eUp = regUp._eUp;
  368. var eLo = regLo._eUp;
  369. if (Geom.VertLeq(eUp._Org, eLo._Org))
  370. {
  371. if (Geom.EdgeSign(eLo._Dst, eUp._Org, eLo._Org) > 0.0f)
  372. {
  373. return false;
  374. }
  375. // eUp.Org appears to be below eLo
  376. if (!Geom.VertEq(eUp._Org, eLo._Org))
  377. {
  378. // Splice eUp._Org into eLo
  379. _mesh.SplitEdge(eLo._Sym);
  380. _mesh.Splice(eUp, eLo._Oprev);
  381. regUp._dirty = regLo._dirty = true;
  382. }
  383. else if (eUp._Org != eLo._Org)
  384. {
  385. // merge the two vertices, discarding eUp.Org
  386. _pq.Remove(eUp._Org._pqHandle);
  387. SpliceMergeVertices(eLo._Oprev, eUp);
  388. }
  389. }
  390. else
  391. {
  392. if (Geom.EdgeSign(eUp._Dst, eLo._Org, eUp._Org) < 0.0f)
  393. {
  394. return false;
  395. }
  396. // eLo.Org appears to be above eUp, so splice eLo.Org into eUp
  397. RegionAbove(regUp)._dirty = regUp._dirty = true;
  398. _mesh.SplitEdge(eUp._Sym);
  399. _mesh.Splice(eLo._Oprev, eUp);
  400. }
  401. return true;
  402. }
  403. /// <summary>
  404. /// Check the upper and lower edge of "regUp", to make sure that the
  405. /// eUp->Dst is above eLo, or eLo->Dst is below eUp (depending on which
  406. /// destination is rightmost).
  407. ///
  408. /// Theoretically, this should always be true. However, splitting an edge
  409. /// into two pieces can change the results of previous tests. For example,
  410. /// suppose at one point we checked eUp and eLo, and decided that eUp->Dst
  411. /// is barely above eLo. Then later, we split eLo into two edges (eg. from
  412. /// a splice operation like this one). This can change the result of
  413. /// the test so that now eUp->Dst is incident to eLo, or barely below it.
  414. /// We must correct this condition to maintain the dictionary invariants
  415. /// (otherwise new edges might get inserted in the wrong place in the
  416. /// dictionary, and bad stuff will happen).
  417. ///
  418. /// We fix the problem by just splicing the offending vertex into the
  419. /// other edge.
  420. /// </summary>
  421. private bool CheckForLeftSplice(ActiveRegion regUp)
  422. {
  423. var regLo = RegionBelow(regUp);
  424. var eUp = regUp._eUp;
  425. var eLo = regLo._eUp;
  426. Debug.Assert(!Geom.VertEq(eUp._Dst, eLo._Dst));
  427. if (Geom.VertLeq(eUp._Dst, eLo._Dst))
  428. {
  429. if (Geom.EdgeSign(eUp._Dst, eLo._Dst, eUp._Org) < 0.0f)
  430. {
  431. return false;
  432. }
  433. // eLo.Dst is above eUp, so splice eLo.Dst into eUp
  434. RegionAbove(regUp)._dirty = regUp._dirty = true;
  435. var e = _mesh.SplitEdge(eUp);
  436. _mesh.Splice(eLo._Sym, e);
  437. e._Lface._inside = regUp._inside;
  438. }
  439. else
  440. {
  441. if (Geom.EdgeSign(eLo._Dst, eUp._Dst, eLo._Org) > 0.0f)
  442. {
  443. return false;
  444. }
  445. // eUp.Dst is below eLo, so splice eUp.Dst into eLo
  446. regUp._dirty = regLo._dirty = true;
  447. var e = _mesh.SplitEdge(eLo);
  448. _mesh.Splice(eUp._Lnext, eLo._Sym);
  449. e._Rface._inside = regUp._inside;
  450. }
  451. return true;
  452. }
  453. /// <summary>
  454. /// Check the upper and lower edges of the given region to see if
  455. /// they intersect. If so, create the intersection and add it
  456. /// to the data structures.
  457. ///
  458. /// Returns TRUE if adding the new intersection resulted in a recursive
  459. /// call to AddRightEdges(); in this case all "dirty" regions have been
  460. /// checked for intersections, and possibly regUp has been deleted.
  461. /// </summary>
  462. private bool CheckForIntersect(ActiveRegion regUp)
  463. {
  464. var regLo = RegionBelow(regUp);
  465. var eUp = regUp._eUp;
  466. var eLo = regLo._eUp;
  467. var orgUp = eUp._Org;
  468. var orgLo = eLo._Org;
  469. var dstUp = eUp._Dst;
  470. var dstLo = eLo._Dst;
  471. Debug.Assert(!Geom.VertEq(dstLo, dstUp));
  472. Debug.Assert(Geom.EdgeSign(dstUp, _event, orgUp) <= 0.0f);
  473. Debug.Assert(Geom.EdgeSign(dstLo, _event, orgLo) >= 0.0f);
  474. Debug.Assert(orgUp != _event && orgLo != _event);
  475. Debug.Assert(!regUp._fixUpperEdge && !regLo._fixUpperEdge);
  476. if( orgUp == orgLo )
  477. {
  478. // right endpoints are the same
  479. return false;
  480. }
  481. var tMinUp = Math.Min(orgUp._t, dstUp._t);
  482. var tMaxLo = Math.Max(orgLo._t, dstLo._t);
  483. if( tMinUp > tMaxLo )
  484. {
  485. // t ranges do not overlap
  486. return false;
  487. }
  488. if (Geom.VertLeq(orgUp, orgLo))
  489. {
  490. if (Geom.EdgeSign( dstLo, orgUp, orgLo ) > 0.0f)
  491. {
  492. return false;
  493. }
  494. }
  495. else
  496. {
  497. if (Geom.EdgeSign( dstUp, orgLo, orgUp ) < 0.0f)
  498. {
  499. return false;
  500. }
  501. }
  502. // At this point the edges intersect, at least marginally
  503. var isect = MeshUtils.Vertex.Create();
  504. Geom.EdgeIntersect(dstUp, orgUp, dstLo, orgLo, isect);
  505. // The following properties are guaranteed:
  506. Debug.Assert(Math.Min(orgUp._t, dstUp._t) <= isect._t);
  507. Debug.Assert(isect._t <= Math.Max(orgLo._t, dstLo._t));
  508. Debug.Assert(Math.Min(dstLo._s, dstUp._s) <= isect._s);
  509. Debug.Assert(isect._s <= Math.Max(orgLo._s, orgUp._s));
  510. if (Geom.VertLeq(isect, _event))
  511. {
  512. // The intersection point lies slightly to the left of the sweep line,
  513. // so move it until it''s slightly to the right of the sweep line.
  514. // (If we had perfect numerical precision, this would never happen
  515. // in the first place). The easiest and safest thing to do is
  516. // replace the intersection by tess._event.
  517. isect._s = _event._s;
  518. isect._t = _event._t;
  519. }
  520. // Similarly, if the computed intersection lies to the right of the
  521. // rightmost origin (which should rarely happen), it can cause
  522. // unbelievable inefficiency on sufficiently degenerate inputs.
  523. // (If you have the test program, try running test54.d with the
  524. // "X zoom" option turned on).
  525. var orgMin = Geom.VertLeq(orgUp, orgLo) ? orgUp : orgLo;
  526. if (Geom.VertLeq(orgMin, isect))
  527. {
  528. isect._s = orgMin._s;
  529. isect._t = orgMin._t;
  530. }
  531. if (Geom.VertEq(isect, orgUp) || Geom.VertEq(isect, orgLo))
  532. {
  533. // Easy case -- intersection at one of the right endpoints
  534. CheckForRightSplice(regUp);
  535. return false;
  536. }
  537. if ( (! Geom.VertEq(dstUp, _event)
  538. && Geom.EdgeSign(dstUp, _event, isect) >= 0.0f)
  539. || (! Geom.VertEq(dstLo, _event)
  540. && Geom.EdgeSign(dstLo, _event, isect) <= 0.0f))
  541. {
  542. // Very unusual -- the new upper or lower edge would pass on the
  543. // wrong side of the sweep event, or through it. This can happen
  544. // due to very small numerical errors in the intersection calculation.
  545. if (dstLo == _event)
  546. {
  547. // Splice dstLo into eUp, and process the new region(s)
  548. _mesh.SplitEdge(eUp._Sym);
  549. _mesh.Splice(eLo._Sym, eUp);
  550. regUp = TopLeftRegion(regUp);
  551. eUp = RegionBelow(regUp)._eUp;
  552. FinishLeftRegions(RegionBelow(regUp), regLo);
  553. AddRightEdges(regUp, eUp._Oprev, eUp, eUp, true);
  554. return true;
  555. }
  556. if( dstUp == _event ) {
  557. /* Splice dstUp into eLo, and process the new region(s) */
  558. _mesh.SplitEdge(eLo._Sym);
  559. _mesh.Splice(eUp._Lnext, eLo._Oprev);
  560. regLo = regUp;
  561. regUp = TopRightRegion(regUp);
  562. var e = RegionBelow(regUp)._eUp._Rprev;
  563. regLo._eUp = eLo._Oprev;
  564. eLo = FinishLeftRegions(regLo, null);
  565. AddRightEdges(regUp, eLo._Onext, eUp._Rprev, e, true);
  566. return true;
  567. }
  568. // Special case: called from ConnectRightVertex. If either
  569. // edge passes on the wrong side of tess._event, split it
  570. // (and wait for ConnectRightVertex to splice it appropriately).
  571. if (Geom.EdgeSign( dstUp, _event, isect ) >= 0.0f)
  572. {
  573. RegionAbove(regUp)._dirty = regUp._dirty = true;
  574. _mesh.SplitEdge(eUp._Sym);
  575. eUp._Org._s = _event._s;
  576. eUp._Org._t = _event._t;
  577. }
  578. if (Geom.EdgeSign(dstLo, _event, isect) <= 0.0f)
  579. {
  580. regUp._dirty = regLo._dirty = true;
  581. _mesh.SplitEdge(eLo._Sym);
  582. eLo._Org._s = _event._s;
  583. eLo._Org._t = _event._t;
  584. }
  585. // leave the rest for ConnectRightVertex
  586. return false;
  587. }
  588. // General case -- split both edges, splice into new vertex.
  589. // When we do the splice operation, the order of the arguments is
  590. // arbitrary as far as correctness goes. However, when the operation
  591. // creates a new face, the work done is proportional to the size of
  592. // the new face. We expect the faces in the processed part of
  593. // the mesh (ie. eUp._Lface) to be smaller than the faces in the
  594. // unprocessed original contours (which will be eLo._Oprev._Lface).
  595. _mesh.SplitEdge(eUp._Sym);
  596. _mesh.SplitEdge(eLo._Sym);
  597. _mesh.Splice(eLo._Oprev, eUp);
  598. eUp._Org._s = isect._s;
  599. eUp._Org._t = isect._t;
  600. eUp._Org._pqHandle = _pq.Insert(eUp._Org);
  601. if (eUp._Org._pqHandle._handle == PQHandle.Invalid)
  602. {
  603. throw new InvalidOperationException("PQHandle should not be invalid");
  604. }
  605. GetIntersectData(eUp._Org, orgUp, dstUp, orgLo, dstLo);
  606. RegionAbove(regUp)._dirty = regUp._dirty = regLo._dirty = true;
  607. return false;
  608. }
  609. /// <summary>
  610. /// When the upper or lower edge of any region changes, the region is
  611. /// marked "dirty". This routine walks through all the dirty regions
  612. /// and makes sure that the dictionary invariants are satisfied
  613. /// (see the comments at the beginning of this file). Of course
  614. /// new dirty regions can be created as we make changes to restore
  615. /// the invariants.
  616. /// </summary>
  617. private void WalkDirtyRegions(ActiveRegion regUp)
  618. {
  619. var regLo = RegionBelow(regUp);
  620. MeshUtils.Edge eUp, eLo;
  621. while (true)
  622. {
  623. // Find the lowest dirty region (we walk from the bottom up).
  624. while (regLo._dirty)
  625. {
  626. regUp = regLo;
  627. regLo = RegionBelow(regLo);
  628. }
  629. if (!regUp._dirty)
  630. {
  631. regLo = regUp;
  632. regUp = RegionAbove( regUp );
  633. if(regUp == null || !regUp._dirty)
  634. {
  635. // We've walked all the dirty regions
  636. return;
  637. }
  638. }
  639. regUp._dirty = false;
  640. eUp = regUp._eUp;
  641. eLo = regLo._eUp;
  642. if (eUp._Dst != eLo._Dst)
  643. {
  644. // Check that the edge ordering is obeyed at the Dst vertices.
  645. if (CheckForLeftSplice(regUp))
  646. {
  647. // If the upper or lower edge was marked fixUpperEdge, then
  648. // we no longer need it (since these edges are needed only for
  649. // vertices which otherwise have no right-going edges).
  650. if (regLo._fixUpperEdge)
  651. {
  652. DeleteRegion(regLo);
  653. _mesh.Delete(eLo);
  654. regLo = RegionBelow(regUp);
  655. eLo = regLo._eUp;
  656. }
  657. else if( regUp._fixUpperEdge )
  658. {
  659. DeleteRegion(regUp);
  660. _mesh.Delete(eUp);
  661. regUp = RegionAbove(regLo);
  662. eUp = regUp._eUp;
  663. }
  664. }
  665. }
  666. if (eUp._Org != eLo._Org)
  667. {
  668. if( eUp._Dst != eLo._Dst
  669. && ! regUp._fixUpperEdge && ! regLo._fixUpperEdge
  670. && (eUp._Dst == _event || eLo._Dst == _event) )
  671. {
  672. // When all else fails in CheckForIntersect(), it uses tess._event
  673. // as the intersection location. To make this possible, it requires
  674. // that tess._event lie between the upper and lower edges, and also
  675. // that neither of these is marked fixUpperEdge (since in the worst
  676. // case it might splice one of these edges into tess.event, and
  677. // violate the invariant that fixable edges are the only right-going
  678. // edge from their associated vertex).
  679. if (CheckForIntersect(regUp))
  680. {
  681. // WalkDirtyRegions() was called recursively; we're done
  682. return;
  683. }
  684. }
  685. else
  686. {
  687. // Even though we can't use CheckForIntersect(), the Org vertices
  688. // may violate the dictionary edge ordering. Check and correct this.
  689. CheckForRightSplice(regUp);
  690. }
  691. }
  692. if (eUp._Org == eLo._Org && eUp._Dst == eLo._Dst)
  693. {
  694. // A degenerate loop consisting of only two edges -- delete it.
  695. Geom.AddWinding(eLo, eUp);
  696. DeleteRegion(regUp);
  697. _mesh.Delete(eUp);
  698. regUp = RegionAbove(regLo);
  699. }
  700. }
  701. }
  702. /// <summary>
  703. /// Purpose: connect a "right" vertex vEvent (one where all edges go left)
  704. /// to the unprocessed portion of the mesh. Since there are no right-going
  705. /// edges, two regions (one above vEvent and one below) are being merged
  706. /// into one. "regUp" is the upper of these two regions.
  707. ///
  708. /// There are two reasons for doing this (adding a right-going edge):
  709. /// - if the two regions being merged are "inside", we must add an edge
  710. /// to keep them separated (the combined region would not be monotone).
  711. /// - in any case, we must leave some record of vEvent in the dictionary,
  712. /// so that we can merge vEvent with features that we have not seen yet.
  713. /// For example, maybe there is a vertical edge which passes just to
  714. /// the right of vEvent; we would like to splice vEvent into this edge.
  715. ///
  716. /// However, we don't want to connect vEvent to just any vertex. We don''t
  717. /// want the new edge to cross any other edges; otherwise we will create
  718. /// intersection vertices even when the input data had no self-intersections.
  719. /// (This is a bad thing; if the user's input data has no intersections,
  720. /// we don't want to generate any false intersections ourselves.)
  721. ///
  722. /// Our eventual goal is to connect vEvent to the leftmost unprocessed
  723. /// vertex of the combined region (the union of regUp and regLo).
  724. /// But because of unseen vertices with all right-going edges, and also
  725. /// new vertices which may be created by edge intersections, we don''t
  726. /// know where that leftmost unprocessed vertex is. In the meantime, we
  727. /// connect vEvent to the closest vertex of either chain, and mark the region
  728. /// as "fixUpperEdge". This flag says to delete and reconnect this edge
  729. /// to the next processed vertex on the boundary of the combined region.
  730. /// Quite possibly the vertex we connected to will turn out to be the
  731. /// closest one, in which case we won''t need to make any changes.
  732. /// </summary>
  733. private void ConnectRightVertex(ActiveRegion regUp, MeshUtils.Edge eBottomLeft)
  734. {
  735. var eTopLeft = eBottomLeft._Onext;
  736. var regLo = RegionBelow(regUp);
  737. var eUp = regUp._eUp;
  738. var eLo = regLo._eUp;
  739. bool degenerate = false;
  740. if (eUp._Dst != eLo._Dst)
  741. {
  742. CheckForIntersect(regUp);
  743. }
  744. // Possible new degeneracies: upper or lower edge of regUp may pass
  745. // through vEvent, or may coincide with new intersection vertex
  746. if (Geom.VertEq(eUp._Org, _event))
  747. {
  748. _mesh.Splice(eTopLeft._Oprev, eUp);
  749. regUp = TopLeftRegion(regUp);
  750. eTopLeft = RegionBelow(regUp)._eUp;
  751. FinishLeftRegions(RegionBelow(regUp), regLo);
  752. degenerate = true;
  753. }
  754. if (Geom.VertEq(eLo._Org, _event))
  755. {
  756. _mesh.Splice(eBottomLeft, eLo._Oprev);
  757. eBottomLeft = FinishLeftRegions(regLo, null);
  758. degenerate = true;
  759. }
  760. if (degenerate)
  761. {
  762. AddRightEdges(regUp, eBottomLeft._Onext, eTopLeft, eTopLeft, true);
  763. return;
  764. }
  765. // Non-degenerate situation -- need to add a temporary, fixable edge.
  766. // Connect to the closer of eLo.Org, eUp.Org.
  767. MeshUtils.Edge eNew;
  768. if (Geom.VertLeq(eLo._Org, eUp._Org))
  769. {
  770. eNew = eLo._Oprev;
  771. }
  772. else
  773. {
  774. eNew = eUp;
  775. }
  776. eNew = _mesh.Connect(eBottomLeft._Lprev, eNew);
  777. // Prevent cleanup, otherwise eNew might disappear before we've even
  778. // had a chance to mark it as a temporary edge.
  779. AddRightEdges(regUp, eNew, eNew._Onext, eNew._Onext, false);
  780. eNew._Sym._activeRegion._fixUpperEdge = true;
  781. WalkDirtyRegions(regUp);
  782. }
  783. /// <summary>
  784. /// The event vertex lies exacty on an already-processed edge or vertex.
  785. /// Adding the new vertex involves splicing it into the already-processed
  786. /// part of the mesh.
  787. /// </summary>
  788. private void ConnectLeftDegenerate(ActiveRegion regUp, MeshUtils.Vertex vEvent)
  789. {
  790. var e = regUp._eUp;
  791. if (Geom.VertEq(e._Org, vEvent))
  792. {
  793. // e.Org is an unprocessed vertex - just combine them, and wait
  794. // for e.Org to be pulled from the queue
  795. // C# : in the C version, there is a flag but it was never implemented
  796. // the vertices are before beginning the tessellation
  797. throw new InvalidOperationException("Vertices should have been merged before");
  798. }
  799. if (!Geom.VertEq(e._Dst, vEvent))
  800. {
  801. // General case -- splice vEvent into edge e which passes through it
  802. _mesh.SplitEdge(e._Sym);
  803. if (regUp._fixUpperEdge)
  804. {
  805. // This edge was fixable -- delete unused portion of original edge
  806. _mesh.Delete(e._Onext);
  807. regUp._fixUpperEdge = false;
  808. }
  809. _mesh.Splice(vEvent._anEdge, e);
  810. SweepEvent(vEvent); // recurse
  811. return;
  812. }
  813. // See above
  814. throw new InvalidOperationException("Vertices should have been merged before");
  815. }
  816. /// <summary>
  817. /// Purpose: connect a "left" vertex (one where both edges go right)
  818. /// to the processed portion of the mesh. Let R be the active region
  819. /// containing vEvent, and let U and L be the upper and lower edge
  820. /// chains of R. There are two possibilities:
  821. ///
  822. /// - the normal case: split R into two regions, by connecting vEvent to
  823. /// the rightmost vertex of U or L lying to the left of the sweep line
  824. ///
  825. /// - the degenerate case: if vEvent is close enough to U or L, we
  826. /// merge vEvent into that edge chain. The subcases are:
  827. /// - merging with the rightmost vertex of U or L
  828. /// - merging with the active edge of U or L
  829. /// - merging with an already-processed portion of U or L
  830. /// </summary>
  831. private void ConnectLeftVertex(MeshUtils.Vertex vEvent)
  832. {
  833. var tmp = new ActiveRegion();
  834. // Get a pointer to the active region containing vEvent
  835. tmp._eUp = vEvent._anEdge._Sym;
  836. var regUp = _dict.Find(tmp).Key;
  837. var regLo = RegionBelow(regUp);
  838. if (regLo == null)
  839. {
  840. // This may happen if the input polygon is coplanar.
  841. return;
  842. }
  843. var eUp = regUp._eUp;
  844. var eLo = regLo._eUp;
  845. // Try merging with U or L first
  846. if (Geom.EdgeSign(eUp._Dst, vEvent, eUp._Org) == 0.0f)
  847. {
  848. ConnectLeftDegenerate(regUp, vEvent);
  849. return;
  850. }
  851. // Connect vEvent to rightmost processed vertex of either chain.
  852. // e._Dst is the vertex that we will connect to vEvent.
  853. var reg = Geom.VertLeq(eLo._Dst, eUp._Dst) ? regUp : regLo;
  854. if (regUp._inside || reg._fixUpperEdge)
  855. {
  856. MeshUtils.Edge eNew;
  857. if (reg == regUp)
  858. {
  859. eNew = _mesh.Connect(vEvent._anEdge._Sym, eUp._Lnext);
  860. }
  861. else
  862. {
  863. eNew = _mesh.Connect(eLo._Dnext, vEvent._anEdge)._Sym;
  864. }
  865. if (reg._fixUpperEdge)
  866. {
  867. FixUpperEdge(reg, eNew);
  868. }
  869. else
  870. {
  871. ComputeWinding(AddRegionBelow(regUp, eNew));
  872. }
  873. SweepEvent(vEvent);
  874. }
  875. else
  876. {
  877. // The new vertex is in a region which does not belong to the polygon.
  878. // We don't need to connect this vertex to the rest of the mesh.
  879. AddRightEdges(regUp, vEvent._anEdge, vEvent._anEdge, null, true);
  880. }
  881. }
  882. /// <summary>
  883. /// Does everything necessary when the sweep line crosses a vertex.
  884. /// Updates the mesh and the edge dictionary.
  885. /// </summary>
  886. private void SweepEvent(MeshUtils.Vertex vEvent)
  887. {
  888. _event = vEvent;
  889. // Check if this vertex is the right endpoint of an edge that is
  890. // already in the dictionary. In this case we don't need to waste
  891. // time searching for the location to insert new edges.
  892. var e = vEvent._anEdge;
  893. while (e._activeRegion == null)
  894. {
  895. e = e._Onext;
  896. if (e == vEvent._anEdge)
  897. {
  898. // All edges go right -- not incident to any processed edges
  899. ConnectLeftVertex(vEvent);
  900. return;
  901. }
  902. }
  903. // Processing consists of two phases: first we "finish" all the
  904. // active regions where both the upper and lower edges terminate
  905. // at vEvent (ie. vEvent is closing off these regions).
  906. // We mark these faces "inside" or "outside" the polygon according
  907. // to their winding number, and delete the edges from the dictionary.
  908. // This takes care of all the left-going edges from vEvent.
  909. var regUp = TopLeftRegion(e._activeRegion);
  910. var reg = RegionBelow(regUp);
  911. var eTopLeft = reg._eUp;
  912. var eBottomLeft = FinishLeftRegions(reg, null);
  913. // Next we process all the right-going edges from vEvent. This
  914. // involves adding the edges to the dictionary, and creating the
  915. // associated "active regions" which record information about the
  916. // regions between adjacent dictionary edges.
  917. if (eBottomLeft._Onext == eTopLeft)
  918. {
  919. // No right-going edges -- add a temporary "fixable" edge
  920. ConnectRightVertex(regUp, eBottomLeft);
  921. }
  922. else
  923. {
  924. AddRightEdges(regUp, eBottomLeft._Onext, eTopLeft, eTopLeft, true);
  925. }
  926. }
  927. /// <summary>
  928. /// Make the sentinel coordinates big enough that they will never be
  929. /// merged with real input features.
  930. ///
  931. /// We add two sentinel edges above and below all other edges,
  932. /// to avoid special cases at the top and bottom.
  933. /// </summary>
  934. private void AddSentinel(Real smin, Real smax, Real t)
  935. {
  936. var e = _mesh.MakeEdge();
  937. e._Org._s = smax;
  938. e._Org._t = t;
  939. e._Dst._s = smin;
  940. e._Dst._t = t;
  941. _event = e._Dst; // initialize it
  942. var reg = new ActiveRegion();
  943. reg._eUp = e;
  944. reg._windingNumber = 0;
  945. reg._inside = false;
  946. reg._fixUpperEdge = false;
  947. reg._sentinel = true;
  948. reg._dirty = false;
  949. reg._nodeUp = _dict.Insert(reg);
  950. }
  951. /// <summary>
  952. /// We maintain an ordering of edge intersections with the sweep line.
  953. /// This order is maintained in a dynamic dictionary.
  954. /// </summary>
  955. private void InitEdgeDict()
  956. {
  957. _dict = new Dict<ActiveRegion>(EdgeLeq);
  958. AddSentinel(-SentinelCoord, SentinelCoord, -SentinelCoord);
  959. AddSentinel(-SentinelCoord, SentinelCoord, +SentinelCoord);
  960. }
  961. private void DoneEdgeDict()
  962. {
  963. int fixedEdges = 0;
  964. ActiveRegion reg;
  965. while ((reg = _dict.Min().Key) != null)
  966. {
  967. // At the end of all processing, the dictionary should contain
  968. // only the two sentinel edges, plus at most one "fixable" edge
  969. // created by ConnectRightVertex().
  970. if (!reg._sentinel)
  971. {
  972. Debug.Assert(reg._fixUpperEdge);
  973. Debug.Assert(++fixedEdges == 1);
  974. }
  975. Debug.Assert(reg._windingNumber == 0);
  976. DeleteRegion(reg);
  977. }
  978. _dict = null;
  979. }
  980. /// <summary>
  981. /// Remove zero-length edges, and contours with fewer than 3 vertices.
  982. /// </summary>
  983. private void RemoveDegenerateEdges()
  984. {
  985. MeshUtils.Edge eHead = _mesh._eHead, e, eNext, eLnext;
  986. for (e = eHead._next; e != eHead; e = eNext)
  987. {
  988. eNext = e._next;
  989. eLnext = e._Lnext;
  990. if (Geom.VertEq(e._Org, e._Dst) && e._Lnext._Lnext != e)
  991. {
  992. // Zero-length edge, contour has at least 3 edges
  993. SpliceMergeVertices(eLnext, e); // deletes e.Org
  994. _mesh.Delete(e); // e is a self-loop
  995. e = eLnext;
  996. eLnext = e._Lnext;
  997. }
  998. if (eLnext._Lnext == e)
  999. {
  1000. // Degenerate contour (one or two edges)
  1001. if (eLnext != e)
  1002. {
  1003. if (eLnext == eNext || eLnext == eNext._Sym)
  1004. {
  1005. eNext = eNext._next;
  1006. }
  1007. _mesh.Delete(eLnext);
  1008. }
  1009. if (e == eNext || e == eNext._Sym)
  1010. {
  1011. eNext = eNext._next;
  1012. }
  1013. _mesh.Delete(e);
  1014. }
  1015. }
  1016. }
  1017. /// <summary>
  1018. /// Insert all vertices into the priority queue which determines the
  1019. /// order in which vertices cross the sweep line.
  1020. /// </summary>
  1021. private void InitPriorityQ()
  1022. {
  1023. MeshUtils.Vertex vHead = _mesh._vHead, v;
  1024. int vertexCount = 0;
  1025. for (v = vHead._next; v != vHead; v = v._next)
  1026. {
  1027. vertexCount++;
  1028. }
  1029. // Make sure there is enough space for sentinels.
  1030. vertexCount += 8;
  1031. _pq = new PriorityQueue<MeshUtils.Vertex>(vertexCount, Geom.VertLeq);
  1032. vHead = _mesh._vHead;
  1033. for( v = vHead._next; v != vHead; v = v._next ) {
  1034. v._pqHandle = _pq.Insert(v);
  1035. if (v._pqHandle._handle == PQHandle.Invalid)
  1036. {
  1037. throw new InvalidOperationException("PQHandle should not be invalid");
  1038. }
  1039. }
  1040. _pq.Init();
  1041. }
  1042. private void DonePriorityQ()
  1043. {
  1044. _pq = null;
  1045. }
  1046. /// <summary>
  1047. /// Delete any degenerate faces with only two edges. WalkDirtyRegions()
  1048. /// will catch almost all of these, but it won't catch degenerate faces
  1049. /// produced by splice operations on already-processed edges.
  1050. /// The two places this can happen are in FinishLeftRegions(), when
  1051. /// we splice in a "temporary" edge produced by ConnectRightVertex(),
  1052. /// and in CheckForLeftSplice(), where we splice already-processed
  1053. /// edges to ensure that our dictionary invariants are not violated
  1054. /// by numerical errors.
  1055. ///
  1056. /// In both these cases it is *very* dangerous to delete the offending
  1057. /// edge at the time, since one of the routines further up the stack
  1058. /// will sometimes be keeping a pointer to that edge.
  1059. /// </summary>
  1060. private void RemoveDegenerateFaces()
  1061. {
  1062. MeshUtils.Face f, fNext;
  1063. MeshUtils.Edge e;
  1064. for (f = _mesh._fHead._next; f != _mesh._fHead; f = fNext)
  1065. {
  1066. fNext = f._next;
  1067. e = f._anEdge;
  1068. Debug.Assert(e._Lnext != e);
  1069. if (e._Lnext._Lnext == e)
  1070. {
  1071. // A face with only two edges
  1072. Geom.AddWinding(e._Onext, e);
  1073. _mesh.Delete(e);
  1074. }
  1075. }
  1076. }
  1077. /// <summary>
  1078. /// ComputeInterior computes the planar arrangement specified
  1079. /// by the given contours, and further subdivides this arrangement
  1080. /// into regions. Each region is marked "inside" if it belongs
  1081. /// to the polygon, according to the rule given by windingRule.
  1082. /// Each interior region is guaranteed to be monotone.
  1083. /// </summary>
  1084. protected void ComputeInterior()
  1085. {
  1086. // Each vertex defines an event for our sweep line. Start by inserting
  1087. // all the vertices in a priority queue. Events are processed in
  1088. // lexicographic order, ie.
  1089. //
  1090. // e1 < e2 iff e1.x < e2.x || (e1.x == e2.x && e1.y < e2.y)
  1091. RemoveDegenerateEdges();
  1092. InitPriorityQ();
  1093. RemoveDegenerateFaces();
  1094. InitEdgeDict();
  1095. MeshUtils.Vertex v, vNext;
  1096. while ((v = _pq.ExtractMin()) != null)
  1097. {
  1098. while (true)
  1099. {
  1100. vNext = _pq.Minimum();
  1101. if (vNext == null || !Geom.VertEq(vNext, v))
  1102. {
  1103. break;
  1104. }
  1105. // Merge together all vertices at exactly the same location.
  1106. // This is more efficient than processing them one at a time,
  1107. // simplifies the code (see ConnectLeftDegenerate), and is also
  1108. // important for correct handling of certain degenerate cases.
  1109. // For example, suppose there are two identical edges A and B
  1110. // that belong to different contours (so without this code they would
  1111. // be processed by separate sweep events). Suppose another edge C
  1112. // crosses A and B from above. When A is processed, we split it
  1113. // at its intersection point with C. However this also splits C,
  1114. // so when we insert B we may compute a slightly different
  1115. // intersection point. This might leave two edges with a small
  1116. // gap between them. This kind of error is especially obvious
  1117. // when using boundary extraction (BoundaryOnly).
  1118. vNext = _pq.ExtractMin();
  1119. SpliceMergeVertices(v._anEdge, vNext._anEdge);
  1120. }
  1121. SweepEvent(v);
  1122. }
  1123. DoneEdgeDict();
  1124. DonePriorityQ();
  1125. RemoveDegenerateFaces();
  1126. _mesh.Check();
  1127. }
  1128. }
  1129. }
  1130. }