mirror of
https://github.com/jhasse/poly2tri.git
synced 2025-01-07 10:23:29 +01:00
ported MonotoneMountain
This commit is contained in:
parent
1c75a2dc1c
commit
696e99da3c
@ -29,7 +29,6 @@
|
|||||||
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
#
|
#
|
||||||
from math import floor
|
|
||||||
|
|
||||||
###
|
###
|
||||||
### Based on Raimund Seidel'e paper "A simple and fast incremental randomized
|
### Based on Raimund Seidel'e paper "A simple and fast incremental randomized
|
||||||
@ -39,6 +38,8 @@ from math import floor
|
|||||||
cdef extern from 'math.h':
|
cdef extern from 'math.h':
|
||||||
double cos(double)
|
double cos(double)
|
||||||
double sin(double)
|
double sin(double)
|
||||||
|
double atan2(double, double)
|
||||||
|
double floor(double)
|
||||||
double sqrt(double)
|
double sqrt(double)
|
||||||
|
|
||||||
cdef list merge_sort(list l):
|
cdef list merge_sort(list l):
|
||||||
@ -494,3 +495,95 @@ class QueryGraph:
|
|||||||
qNode = XNode(e.q, yNode, Sink(tlist[2]))
|
qNode = XNode(e.q, yNode, Sink(tlist[2]))
|
||||||
self.replace(sink, qNode)
|
self.replace(sink, qNode)
|
||||||
|
|
||||||
|
cdef float PI_SLOP = 3.1
|
||||||
|
|
||||||
|
cdef class MonotoneMountain:
|
||||||
|
|
||||||
|
cdef:
|
||||||
|
Point tail, head
|
||||||
|
int size
|
||||||
|
list convex_points
|
||||||
|
list mono_poly
|
||||||
|
list triangles
|
||||||
|
list convex_polies
|
||||||
|
bool positive
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.size = 0
|
||||||
|
self.tail, self.head = None
|
||||||
|
self.positive = False
|
||||||
|
self.convex_points = []
|
||||||
|
self.mono_poly = []
|
||||||
|
self.triangles = []
|
||||||
|
self.convex_polies = []
|
||||||
|
|
||||||
|
def append(self, Point point):
|
||||||
|
if self.size == 0:
|
||||||
|
self.head = point
|
||||||
|
self.size += 1
|
||||||
|
elif self.size == 1:
|
||||||
|
if point.not_equal(self.head):
|
||||||
|
self.tail = point
|
||||||
|
self.tail.prev = self.head
|
||||||
|
self.head.next = self.tail
|
||||||
|
self.size += 1
|
||||||
|
else:
|
||||||
|
if point.not_equal(self.tail):
|
||||||
|
self.tail.next = point
|
||||||
|
point.prev = self.tail
|
||||||
|
self.tail = point
|
||||||
|
self.size += 1
|
||||||
|
|
||||||
|
def remove(self, Point point):
|
||||||
|
next = point.next
|
||||||
|
prev = point.prev
|
||||||
|
point.prev.next = next
|
||||||
|
point.next.prev = prev
|
||||||
|
self.size -= 1
|
||||||
|
|
||||||
|
def process(self):
|
||||||
|
self.positive = self.angle_sign()
|
||||||
|
self.gen_mono_poly()
|
||||||
|
p = self.head.next
|
||||||
|
while p != self.tail:
|
||||||
|
a = self.angle(p)
|
||||||
|
if a >= PI_SLOP or a <= -PI_SLOP: self.remove(p)
|
||||||
|
elif self.is_convex(p): self.convex_points.append(p)
|
||||||
|
p = p.next
|
||||||
|
self.triangulate()
|
||||||
|
|
||||||
|
def triangulate(self):
|
||||||
|
while not len(self.convex_points) > 0:
|
||||||
|
ear = self.convex_points.remove(0)
|
||||||
|
a = ear.prev
|
||||||
|
b = ear
|
||||||
|
c = ear.next
|
||||||
|
triangle = [a, b, c]
|
||||||
|
self.triangles.append(triangle)
|
||||||
|
self.remove(ear)
|
||||||
|
if self.valid(a): self.convex_points.append(a)
|
||||||
|
if self.valid(c): self.convex_points.append(c)
|
||||||
|
assert(self.size <= 3, "Triangulation bug, please report")
|
||||||
|
|
||||||
|
def valid(self, Point p):
|
||||||
|
return p != self.head and p != self.tail and self.is_convex(p)
|
||||||
|
|
||||||
|
def gen_mono_poly(self):
|
||||||
|
p = self.head
|
||||||
|
while(p is not None):
|
||||||
|
self.mono_poly.append(p)
|
||||||
|
p = p.next
|
||||||
|
|
||||||
|
def angle(self, Point p):
|
||||||
|
cdef Point a = p.next - p
|
||||||
|
cdef Point b = p.prev - p
|
||||||
|
return atan2(a.cross(b), a.dot(b))
|
||||||
|
|
||||||
|
def angle_sign(self):
|
||||||
|
a = self.head.next - self.head
|
||||||
|
b = self.tail - self.head
|
||||||
|
return atan2(a.cross(b), a.dot(b)) >= 0
|
||||||
|
|
||||||
|
def is_convex(self, Point p):
|
||||||
|
if self.positive != (self.angle(p) >= 0): return False
|
||||||
|
return True
|
||||||
|
Loading…
Reference in New Issue
Block a user