2014-08-30 12:11:25 +00:00
|
|
|
from kivy.app import App
|
|
|
|
from kivy.graphics import Mesh, Color
|
|
|
|
from kivy.graphics.tesselator import Tesselator, WINDING_ODD, TYPE_POLYGONS
|
|
|
|
from kivy.uix.widget import Widget
|
|
|
|
from kivy.uix.floatlayout import FloatLayout
|
|
|
|
from kivy.lang import Builder
|
|
|
|
|
|
|
|
Builder.load_string("""
|
|
|
|
<ShapeBuilder>:
|
|
|
|
BoxLayout:
|
|
|
|
size_hint_y: None
|
|
|
|
height: "48dp"
|
|
|
|
spacing: "2dp"
|
|
|
|
padding: "2dp"
|
|
|
|
|
|
|
|
ToggleButton:
|
|
|
|
text: "Debug"
|
|
|
|
id: debug
|
|
|
|
on_release: root.build()
|
|
|
|
Button:
|
|
|
|
text: "New shape"
|
|
|
|
on_release: root.push_shape()
|
|
|
|
Button:
|
|
|
|
text: "Build"
|
|
|
|
on_release: root.build()
|
|
|
|
Button:
|
|
|
|
text: "Reset"
|
|
|
|
on_release: root.reset()
|
|
|
|
|
|
|
|
BoxLayout:
|
|
|
|
size_hint_y: None
|
|
|
|
height: "48dp"
|
|
|
|
top: root.top
|
|
|
|
spacing: "2dp"
|
|
|
|
padding: "2dp"
|
|
|
|
Label:
|
|
|
|
id: status
|
|
|
|
text: "Status"
|
|
|
|
""")
|
|
|
|
|
|
|
|
|
|
|
|
class ShapeBuilder(FloatLayout):
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(ShapeBuilder, self).__init__(**kwargs)
|
|
|
|
self.shapes = [
|
2014-08-30 15:02:34 +00:00
|
|
|
[100, 100, 300, 100, 300, 300, 100, 300],
|
|
|
|
[150, 150, 250, 150, 250, 250, 150, 250]
|
2014-08-30 12:11:25 +00:00
|
|
|
]
|
|
|
|
self.shape = []
|
|
|
|
self.build()
|
|
|
|
|
|
|
|
def on_touch_down(self, touch):
|
|
|
|
if super(ShapeBuilder, self).on_touch_down(touch):
|
|
|
|
return True
|
2014-08-30 15:21:11 +00:00
|
|
|
print(touch.pos)
|
2014-08-30 12:11:25 +00:00
|
|
|
self.shape.extend(touch.pos)
|
|
|
|
self.build()
|
|
|
|
return True
|
|
|
|
|
|
|
|
def on_touch_move(self, touch):
|
|
|
|
if super(ShapeBuilder, self).on_touch_move(touch):
|
|
|
|
return True
|
2014-08-30 15:21:11 +00:00
|
|
|
print(touch.pos)
|
2014-08-30 12:11:25 +00:00
|
|
|
self.shape.extend(touch.pos)
|
|
|
|
self.build()
|
|
|
|
return True
|
|
|
|
|
|
|
|
def on_touch_up(self, touch):
|
|
|
|
if super(ShapeBuilder, self).on_touch_up(touch):
|
|
|
|
return True
|
|
|
|
self.push_shape()
|
|
|
|
self.build()
|
|
|
|
|
|
|
|
def push_shape(self):
|
|
|
|
self.shapes.append(self.shape)
|
|
|
|
self.shape = []
|
|
|
|
|
|
|
|
def build(self):
|
|
|
|
tess = Tesselator()
|
|
|
|
count = 0
|
|
|
|
for shape in self.shapes:
|
|
|
|
if len(shape) >= 3:
|
|
|
|
tess.add_contour(shape)
|
|
|
|
count += 1
|
|
|
|
if self.shape and len(self.shape) >= 3:
|
|
|
|
tess.add_contour(self.shape)
|
|
|
|
count += 1
|
|
|
|
if not count:
|
|
|
|
return
|
2014-08-30 15:21:11 +00:00
|
|
|
print("Tesselate {} shapes".format(count))
|
2014-08-30 12:11:25 +00:00
|
|
|
ret = tess.tesselate(WINDING_ODD, TYPE_POLYGONS)
|
2014-08-30 15:21:11 +00:00
|
|
|
print("Result: {}".format(ret))
|
|
|
|
print("Vertex count: {}".format(tess.vertex_count))
|
|
|
|
print("Element count: {}".format(tess.element_count))
|
2014-08-30 12:11:25 +00:00
|
|
|
|
|
|
|
self.canvas.after.clear()
|
|
|
|
|
|
|
|
debug = self.ids.debug.state == "down"
|
|
|
|
if debug:
|
|
|
|
from random import random
|
|
|
|
with self.canvas.after:
|
|
|
|
c = 0
|
2014-08-30 15:02:34 +00:00
|
|
|
for vertices, indices in tess.meshes:
|
2014-08-30 12:11:25 +00:00
|
|
|
Color(c, 1, 1, mode="hsv")
|
2014-08-30 15:02:34 +00:00
|
|
|
c += 0.3
|
2014-08-30 12:11:25 +00:00
|
|
|
indices = [0]
|
|
|
|
for i in range(1, len(vertices) / 4):
|
|
|
|
if i > 0:
|
|
|
|
indices.append(i)
|
|
|
|
indices.append(i)
|
|
|
|
indices.append(0)
|
|
|
|
indices.append(i)
|
|
|
|
indices.pop(-1)
|
|
|
|
Mesh(vertices=vertices, indices=indices, mode="lines")
|
|
|
|
else:
|
|
|
|
with self.canvas.after:
|
|
|
|
Color(1, 1, 1, 1)
|
2014-08-30 15:02:34 +00:00
|
|
|
for vertices, indices in tess.meshes:
|
2014-08-30 12:11:25 +00:00
|
|
|
Mesh(vertices=vertices, indices=indices, mode="triangle_fan")
|
|
|
|
|
|
|
|
self.ids.status.text = "Vertex: {} - Elements: {}".format(
|
|
|
|
tess.vertex_count, tess.element_count)
|
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
self.shapes = []
|
|
|
|
self.shape = []
|
|
|
|
self.canvas.after.clear()
|
|
|
|
|
|
|
|
|
|
|
|
class TessApp(App):
|
|
|
|
def build(self):
|
|
|
|
return ShapeBuilder()
|
|
|
|
|
|
|
|
TessApp().run()
|