Module sverchok.utils.csg_geom
Expand source code
import math
class CSGVector(object):
"""
class CSGVector
Represents a 3D vector.
Example usage:
CSGVector(1, 2, 3);
"""
def __init__(self, *args):
self.x = args[0]
self.y = args[1]
self.z = args[2]
def clone(self):
return CSGVector(self.x, self.y, self.z)
def negated(self):
return CSGVector(-self.x, -self.y, -self.z)
def plus(self, a):
return CSGVector(self.x + a.x, self.y + a.y, self.z + a.z)
def minus(self, a):
return CSGVector(self.x - a.x, self.y - a.y, self.z - a.z)
def times(self, a):
return CSGVector(self.x * a, self.y * a, self.z * a)
def dividedBy(self, a):
return CSGVector(self.x / a, self.y / a, self.z / a)
def dot(self, a):
return self.x * a.x + self.y * a.y + self.z * a.z
def lerp(self, a, t):
return self.plus(a.minus(self).times(t))
def length(self):
return math.sqrt(self.dot(self))
def unit(self):
""" Normalize. """
return self.dividedBy(self.length())
def cross(self, a):
return CSGVector(
self.y * a.z - self.z * a.y,
self.z * a.x - self.x * a.z,
self.x * a.y - self.y * a.x)
def __getitem__(self, key):
return (self.x, self.y, self.z)[key]
def __setitem__(self, key, value):
l = [self.x, self.y, self.z]
l[key] = value
self.x, self.y, self.z = l
def __len__(self):
return 3
def __iter__(self):
return iter((self.x, self.y, self.z))
def __repr__(self):
return 'CSGVector(%.2f, %.2f, %0.2f)' % (self.x, self.y, self.z)
class CSGVertex(object):
"""
Class CSGVertex
Represents a vertex of a polygon. Use your own vertex class instead of this
one to provide additional features like texture coordinates and vertex
colors. Custom vertex classes need to provide a `pos` property and `clone()`,
`flip()`, and `interpolate()` methods that behave analogous to the ones
defined by `Vertex`. This class provides `normal` so convenience
functions like `CSG.sphere()` can return a smooth vertex normal, but `normal`
is not used anywhere else.
"""
def __init__(self, pos, normal=[0.0, 0.0, 0.0]):
self.pos = CSGVector(pos[0], pos[1], pos[2])
self.normal = CSGVector(normal[0], normal[1], normal[2])
def clone(self):
return CSGVertex(self.pos.clone(), self.normal.clone())
def flip(self):
"""
Invert all orientation-specific data (e.g. vertex normal). Called when the
orientation of a polygon is flipped.
"""
self.normal = self.normal.negated()
def interpolate(self, other, t):
"""
Create a new vertex between this vertex and `other` by linearly
interpolating all properties using a parameter of `t`. Subclasses should
override this to interpolate additional properties.
"""
return CSGVertex(self.pos.lerp(other.pos, t), self.normal.lerp(other.normal, t))
class CSGPlane(object):
"""
class CSGPlane
Represents a plane in 3D space.
"""
"""
`CSGPlane.EPSILON` is the tolerance used by `splitPolygon()` to decide if a
point is on the plane.
"""
EPSILON = 1e-5
def __init__(self, normal, w):
self.normal = normal
self.w = w
@classmethod
def fromPoints(cls, a, b, c):
n = b.minus(a).cross(c.minus(a)).unit()
return CSGPlane(n, n.dot(a))
def clone(self):
return CSGPlane(self.normal.clone(), self.w)
def flip(self):
self.normal = self.normal.negated()
self.w = -self.w
def splitPolygon(self, polygon, coplanarFront, coplanarBack, front, back):
"""
Split `polygon` by this plane if needed, then put the polygon or polygon
fragments in the appropriate lists. Coplanar polygons go into either
`coplanarFront` or `coplanarBack` depending on their orientation with
respect to this plane. Polygons in front or in back of this plane go into
either `front` or `back`
"""
COPLANAR = 0
FRONT = 1
BACK = 2
SPANNING = 3
# Classify each point as well as the entire polygon into one of the above
# four classes.
polygonType = 0
types = []
for i in range(0, len(polygon.vertices)):
t = self.normal.dot(polygon.vertices[i].pos) - self.w
type = -1
if t < -CSGPlane.EPSILON:
type = BACK
elif t > CSGPlane.EPSILON:
type = FRONT
else:
type = COPLANAR
polygonType |= type
types.append(type)
# Put the polygon in the correct list, splitting it when necessary.
if polygonType == COPLANAR:
if self.normal.dot(polygon.plane.normal) > 0:
coplanarFront.append(polygon)
else:
coplanarBack.append(polygon)
elif polygonType == FRONT:
front.append(polygon)
elif polygonType == BACK:
back.append(polygon)
elif polygonType == SPANNING:
f = []
b = []
for i in range(0, len(polygon.vertices)):
j = (i + 1) % len(polygon.vertices)
ti = types[i]
tj = types[j]
vi = polygon.vertices[i]
vj = polygon.vertices[j]
if ti != BACK:
f.append(vi)
if ti != FRONT:
if ti != BACK:
b.append(vi.clone())
else:
b.append(vi)
if (ti | tj) == SPANNING:
t = (self.w - self.normal.dot(vi.pos)) / self.normal.dot(vj.pos.minus(vi.pos))
v = vi.interpolate(vj, t)
f.append(v)
b.append(v.clone())
if len(f) >= 3:
front.append(CSGPolygon(f, polygon.shared))
if len(b) >= 3:
back.append(CSGPolygon(b, polygon.shared))
class CSGPolygon(object):
"""
class CSGPolygon
Represents a convex polygon. The vertices used to initialize a polygon must
be coplanar and form a convex loop. They do not have to be `Vertex`
instances but they must behave similarly (duck typing can be used for
customization).
Each convex polygon has a `shared` property, which is shared between all
polygons that are clones of each other or were split from the same polygon.
This can be used to define per-polygon properties (such as surface color).
"""
def __init__(self, vertices, shared=None):
self.vertices = list(vertices)
self.shared = shared
self.plane = CSGPlane.fromPoints(
self.vertices[0].pos,
self.vertices[1].pos,
self.vertices[2].pos)
def clone(self):
vertices = map(lambda v: v.clone(), self.vertices)
return CSGPolygon(vertices, self.shared)
def flip(self):
self.vertices.reverse()
map(lambda v: v.flip(), self.vertices)
self.plane.flip()
class CSGNode(object):
"""
class CSGNode
Holds a node in a BSP tree. A BSP tree is built from a collection of polygons
by picking a polygon to split along. That polygon (and all other coplanar
polygons) are added directly to that node and the other polygons are added to
the front and/or back subtrees. This is not a leafy BSP tree since there is
no distinction between internal and leaf nodes.
"""
def __init__(self, polygons=None):
self.plane = None
self.front = None
self.back = None
self.polygons = []
if polygons:
self.build(polygons)
def clone(self):
node = CSGNode()
if self.plane:
node.plane = self.plane.clone()
if self.front:
node.front = self.front.clone()
if self.back:
node.back = self.back.clone()
node.polygons = map(lambda p: p.clone(), self.polygons)
return node
def invert(self):
"""
Convert solid space to empty space and empty space to solid space.
"""
for poly in self.polygons:
poly.flip()
if self.plane:
self.plane.flip()
if self.front:
self.front.invert()
if self.back:
self.back.invert()
temp = self.front
self.front = self.back
self.back = temp
def clipPolygons(self, polygons):
"""
Recursively remove all polygons in `polygons` that are inside this BSP
tree.
"""
if not self.plane:
return polygons[:]
front = []
back = []
for poly in polygons:
self.plane.splitPolygon(poly, front, back, front, back)
if self.front:
front = self.front.clipPolygons(front)
if self.back:
back = self.back.clipPolygons(back)
else:
back = []
front.extend(back)
return front
def clipTo(self, bsp):
"""
Remove all polygons in this BSP tree that are inside the other BSP tree
`bsp`.
"""
self.polygons = bsp.clipPolygons(self.polygons)
if self.front:
self.front.clipTo(bsp)
if self.back:
self.back.clipTo(bsp)
def allPolygons(self):
"""
Return a list of all polygons in this BSP tree.
"""
polygons = self.polygons[:]
if self.front:
polygons.extend(self.front.allPolygons())
if self.back:
polygons.extend(self.back.allPolygons())
return polygons
def build(self, polygons):
if isinstance(polygons, map):
polygons = list(polygons)
if not len(polygons):
return
if not self.plane:
self.plane = polygons[0].plane.clone()
front = []
back = []
for poly in polygons:
self.plane.splitPolygon(
poly, self.polygons, self.polygons, front, back)
if len(front):
if not self.front:
self.front = CSGNode()
self.front.build(front)
if len(back):
if not self.back:
self.back = CSGNode()
self.back.build(back)
Classes
class CSGNode (polygons=None)
-
class CSGNode
Holds a node in a BSP tree. A BSP tree is built from a collection of polygons by picking a polygon to split along. That polygon (and all other coplanar polygons) are added directly to that node and the other polygons are added to the front and/or back subtrees. This is not a leafy BSP tree since there is no distinction between internal and leaf nodes.
Expand source code
class CSGNode(object): """ class CSGNode Holds a node in a BSP tree. A BSP tree is built from a collection of polygons by picking a polygon to split along. That polygon (and all other coplanar polygons) are added directly to that node and the other polygons are added to the front and/or back subtrees. This is not a leafy BSP tree since there is no distinction between internal and leaf nodes. """ def __init__(self, polygons=None): self.plane = None self.front = None self.back = None self.polygons = [] if polygons: self.build(polygons) def clone(self): node = CSGNode() if self.plane: node.plane = self.plane.clone() if self.front: node.front = self.front.clone() if self.back: node.back = self.back.clone() node.polygons = map(lambda p: p.clone(), self.polygons) return node def invert(self): """ Convert solid space to empty space and empty space to solid space. """ for poly in self.polygons: poly.flip() if self.plane: self.plane.flip() if self.front: self.front.invert() if self.back: self.back.invert() temp = self.front self.front = self.back self.back = temp def clipPolygons(self, polygons): """ Recursively remove all polygons in `polygons` that are inside this BSP tree. """ if not self.plane: return polygons[:] front = [] back = [] for poly in polygons: self.plane.splitPolygon(poly, front, back, front, back) if self.front: front = self.front.clipPolygons(front) if self.back: back = self.back.clipPolygons(back) else: back = [] front.extend(back) return front def clipTo(self, bsp): """ Remove all polygons in this BSP tree that are inside the other BSP tree `bsp`. """ self.polygons = bsp.clipPolygons(self.polygons) if self.front: self.front.clipTo(bsp) if self.back: self.back.clipTo(bsp) def allPolygons(self): """ Return a list of all polygons in this BSP tree. """ polygons = self.polygons[:] if self.front: polygons.extend(self.front.allPolygons()) if self.back: polygons.extend(self.back.allPolygons()) return polygons def build(self, polygons): if isinstance(polygons, map): polygons = list(polygons) if not len(polygons): return if not self.plane: self.plane = polygons[0].plane.clone() front = [] back = [] for poly in polygons: self.plane.splitPolygon( poly, self.polygons, self.polygons, front, back) if len(front): if not self.front: self.front = CSGNode() self.front.build(front) if len(back): if not self.back: self.back = CSGNode() self.back.build(back)
Methods
def allPolygons(self)
-
Return a list of all polygons in this BSP tree.
Expand source code
def allPolygons(self): """ Return a list of all polygons in this BSP tree. """ polygons = self.polygons[:] if self.front: polygons.extend(self.front.allPolygons()) if self.back: polygons.extend(self.back.allPolygons()) return polygons
def build(self, polygons)
-
Expand source code
def build(self, polygons): if isinstance(polygons, map): polygons = list(polygons) if not len(polygons): return if not self.plane: self.plane = polygons[0].plane.clone() front = [] back = [] for poly in polygons: self.plane.splitPolygon( poly, self.polygons, self.polygons, front, back) if len(front): if not self.front: self.front = CSGNode() self.front.build(front) if len(back): if not self.back: self.back = CSGNode() self.back.build(back)
def clipPolygons(self, polygons)
-
Recursively remove all polygons in
polygons
that are inside this BSP tree.Expand source code
def clipPolygons(self, polygons): """ Recursively remove all polygons in `polygons` that are inside this BSP tree. """ if not self.plane: return polygons[:] front = [] back = [] for poly in polygons: self.plane.splitPolygon(poly, front, back, front, back) if self.front: front = self.front.clipPolygons(front) if self.back: back = self.back.clipPolygons(back) else: back = [] front.extend(back) return front
def clipTo(self, bsp)
-
Remove all polygons in this BSP tree that are inside the other BSP tree
bsp
.Expand source code
def clipTo(self, bsp): """ Remove all polygons in this BSP tree that are inside the other BSP tree `bsp`. """ self.polygons = bsp.clipPolygons(self.polygons) if self.front: self.front.clipTo(bsp) if self.back: self.back.clipTo(bsp)
def clone(self)
-
Expand source code
def clone(self): node = CSGNode() if self.plane: node.plane = self.plane.clone() if self.front: node.front = self.front.clone() if self.back: node.back = self.back.clone() node.polygons = map(lambda p: p.clone(), self.polygons) return node
def invert(self)
-
Convert solid space to empty space and empty space to solid space.
Expand source code
def invert(self): """ Convert solid space to empty space and empty space to solid space. """ for poly in self.polygons: poly.flip() if self.plane: self.plane.flip() if self.front: self.front.invert() if self.back: self.back.invert() temp = self.front self.front = self.back self.back = temp
class CSGPlane (normal, w)
-
class CSGPlane
Represents a plane in 3D space.
Expand source code
class CSGPlane(object): """ class CSGPlane Represents a plane in 3D space. """ """ `CSGPlane.EPSILON` is the tolerance used by `splitPolygon()` to decide if a point is on the plane. """ EPSILON = 1e-5 def __init__(self, normal, w): self.normal = normal self.w = w @classmethod def fromPoints(cls, a, b, c): n = b.minus(a).cross(c.minus(a)).unit() return CSGPlane(n, n.dot(a)) def clone(self): return CSGPlane(self.normal.clone(), self.w) def flip(self): self.normal = self.normal.negated() self.w = -self.w def splitPolygon(self, polygon, coplanarFront, coplanarBack, front, back): """ Split `polygon` by this plane if needed, then put the polygon or polygon fragments in the appropriate lists. Coplanar polygons go into either `coplanarFront` or `coplanarBack` depending on their orientation with respect to this plane. Polygons in front or in back of this plane go into either `front` or `back` """ COPLANAR = 0 FRONT = 1 BACK = 2 SPANNING = 3 # Classify each point as well as the entire polygon into one of the above # four classes. polygonType = 0 types = [] for i in range(0, len(polygon.vertices)): t = self.normal.dot(polygon.vertices[i].pos) - self.w type = -1 if t < -CSGPlane.EPSILON: type = BACK elif t > CSGPlane.EPSILON: type = FRONT else: type = COPLANAR polygonType |= type types.append(type) # Put the polygon in the correct list, splitting it when necessary. if polygonType == COPLANAR: if self.normal.dot(polygon.plane.normal) > 0: coplanarFront.append(polygon) else: coplanarBack.append(polygon) elif polygonType == FRONT: front.append(polygon) elif polygonType == BACK: back.append(polygon) elif polygonType == SPANNING: f = [] b = [] for i in range(0, len(polygon.vertices)): j = (i + 1) % len(polygon.vertices) ti = types[i] tj = types[j] vi = polygon.vertices[i] vj = polygon.vertices[j] if ti != BACK: f.append(vi) if ti != FRONT: if ti != BACK: b.append(vi.clone()) else: b.append(vi) if (ti | tj) == SPANNING: t = (self.w - self.normal.dot(vi.pos)) / self.normal.dot(vj.pos.minus(vi.pos)) v = vi.interpolate(vj, t) f.append(v) b.append(v.clone()) if len(f) >= 3: front.append(CSGPolygon(f, polygon.shared)) if len(b) >= 3: back.append(CSGPolygon(b, polygon.shared))
Class variables
var EPSILON
Static methods
def fromPoints(a, b, c)
-
Expand source code
@classmethod def fromPoints(cls, a, b, c): n = b.minus(a).cross(c.minus(a)).unit() return CSGPlane(n, n.dot(a))
Methods
def clone(self)
-
Expand source code
def clone(self): return CSGPlane(self.normal.clone(), self.w)
def flip(self)
-
Expand source code
def flip(self): self.normal = self.normal.negated() self.w = -self.w
def splitPolygon(self, polygon, coplanarFront, coplanarBack, front, back)
-
Split
polygon
by this plane if needed, then put the polygon or polygon fragments in the appropriate lists. Coplanar polygons go into eithercoplanarFront
orcoplanarBack
depending on their orientation with respect to this plane. Polygons in front or in back of this plane go into eitherfront
orback
Expand source code
def splitPolygon(self, polygon, coplanarFront, coplanarBack, front, back): """ Split `polygon` by this plane if needed, then put the polygon or polygon fragments in the appropriate lists. Coplanar polygons go into either `coplanarFront` or `coplanarBack` depending on their orientation with respect to this plane. Polygons in front or in back of this plane go into either `front` or `back` """ COPLANAR = 0 FRONT = 1 BACK = 2 SPANNING = 3 # Classify each point as well as the entire polygon into one of the above # four classes. polygonType = 0 types = [] for i in range(0, len(polygon.vertices)): t = self.normal.dot(polygon.vertices[i].pos) - self.w type = -1 if t < -CSGPlane.EPSILON: type = BACK elif t > CSGPlane.EPSILON: type = FRONT else: type = COPLANAR polygonType |= type types.append(type) # Put the polygon in the correct list, splitting it when necessary. if polygonType == COPLANAR: if self.normal.dot(polygon.plane.normal) > 0: coplanarFront.append(polygon) else: coplanarBack.append(polygon) elif polygonType == FRONT: front.append(polygon) elif polygonType == BACK: back.append(polygon) elif polygonType == SPANNING: f = [] b = [] for i in range(0, len(polygon.vertices)): j = (i + 1) % len(polygon.vertices) ti = types[i] tj = types[j] vi = polygon.vertices[i] vj = polygon.vertices[j] if ti != BACK: f.append(vi) if ti != FRONT: if ti != BACK: b.append(vi.clone()) else: b.append(vi) if (ti | tj) == SPANNING: t = (self.w - self.normal.dot(vi.pos)) / self.normal.dot(vj.pos.minus(vi.pos)) v = vi.interpolate(vj, t) f.append(v) b.append(v.clone()) if len(f) >= 3: front.append(CSGPolygon(f, polygon.shared)) if len(b) >= 3: back.append(CSGPolygon(b, polygon.shared))
class CSGPolygon (vertices, shared=None)
-
class CSGPolygon
Represents a convex polygon. The vertices used to initialize a polygon must be coplanar and form a convex loop. They do not have to be
Vertex
instances but they must behave similarly (duck typing can be used for customization).Each convex polygon has a
shared
property, which is shared between all polygons that are clones of each other or were split from the same polygon. This can be used to define per-polygon properties (such as surface color).Expand source code
class CSGPolygon(object): """ class CSGPolygon Represents a convex polygon. The vertices used to initialize a polygon must be coplanar and form a convex loop. They do not have to be `Vertex` instances but they must behave similarly (duck typing can be used for customization). Each convex polygon has a `shared` property, which is shared between all polygons that are clones of each other or were split from the same polygon. This can be used to define per-polygon properties (such as surface color). """ def __init__(self, vertices, shared=None): self.vertices = list(vertices) self.shared = shared self.plane = CSGPlane.fromPoints( self.vertices[0].pos, self.vertices[1].pos, self.vertices[2].pos) def clone(self): vertices = map(lambda v: v.clone(), self.vertices) return CSGPolygon(vertices, self.shared) def flip(self): self.vertices.reverse() map(lambda v: v.flip(), self.vertices) self.plane.flip()
Methods
def clone(self)
-
Expand source code
def clone(self): vertices = map(lambda v: v.clone(), self.vertices) return CSGPolygon(vertices, self.shared)
def flip(self)
-
Expand source code
def flip(self): self.vertices.reverse() map(lambda v: v.flip(), self.vertices) self.plane.flip()
class CSGVector (*args)
-
class CSGVector
Represents a 3D vector.
Example usage: CSGVector(1, 2, 3);
Expand source code
class CSGVector(object): """ class CSGVector Represents a 3D vector. Example usage: CSGVector(1, 2, 3); """ def __init__(self, *args): self.x = args[0] self.y = args[1] self.z = args[2] def clone(self): return CSGVector(self.x, self.y, self.z) def negated(self): return CSGVector(-self.x, -self.y, -self.z) def plus(self, a): return CSGVector(self.x + a.x, self.y + a.y, self.z + a.z) def minus(self, a): return CSGVector(self.x - a.x, self.y - a.y, self.z - a.z) def times(self, a): return CSGVector(self.x * a, self.y * a, self.z * a) def dividedBy(self, a): return CSGVector(self.x / a, self.y / a, self.z / a) def dot(self, a): return self.x * a.x + self.y * a.y + self.z * a.z def lerp(self, a, t): return self.plus(a.minus(self).times(t)) def length(self): return math.sqrt(self.dot(self)) def unit(self): """ Normalize. """ return self.dividedBy(self.length()) def cross(self, a): return CSGVector( self.y * a.z - self.z * a.y, self.z * a.x - self.x * a.z, self.x * a.y - self.y * a.x) def __getitem__(self, key): return (self.x, self.y, self.z)[key] def __setitem__(self, key, value): l = [self.x, self.y, self.z] l[key] = value self.x, self.y, self.z = l def __len__(self): return 3 def __iter__(self): return iter((self.x, self.y, self.z)) def __repr__(self): return 'CSGVector(%.2f, %.2f, %0.2f)' % (self.x, self.y, self.z)
Methods
def clone(self)
-
Expand source code
def clone(self): return CSGVector(self.x, self.y, self.z)
def cross(self, a)
-
Expand source code
def cross(self, a): return CSGVector( self.y * a.z - self.z * a.y, self.z * a.x - self.x * a.z, self.x * a.y - self.y * a.x)
def dividedBy(self, a)
-
Expand source code
def dividedBy(self, a): return CSGVector(self.x / a, self.y / a, self.z / a)
def dot(self, a)
-
Expand source code
def dot(self, a): return self.x * a.x + self.y * a.y + self.z * a.z
def length(self)
-
Expand source code
def length(self): return math.sqrt(self.dot(self))
def lerp(self, a, t)
-
Expand source code
def lerp(self, a, t): return self.plus(a.minus(self).times(t))
def minus(self, a)
-
Expand source code
def minus(self, a): return CSGVector(self.x - a.x, self.y - a.y, self.z - a.z)
def negated(self)
-
Expand source code
def negated(self): return CSGVector(-self.x, -self.y, -self.z)
def plus(self, a)
-
Expand source code
def plus(self, a): return CSGVector(self.x + a.x, self.y + a.y, self.z + a.z)
def times(self, a)
-
Expand source code
def times(self, a): return CSGVector(self.x * a, self.y * a, self.z * a)
def unit(self)
-
Normalize.
Expand source code
def unit(self): """ Normalize. """ return self.dividedBy(self.length())
class CSGVertex (pos, normal=[0.0, 0.0, 0.0])
-
Class CSGVertex
Represents a vertex of a polygon. Use your own vertex class instead of this one to provide additional features like texture coordinates and vertex colors. Custom vertex classes need to provide a
pos
property andclone()
,flip()
, andinterpolate()
methods that behave analogous to the ones defined byVertex
. This class providesnormal
so convenience functions likeCSG.sphere()
can return a smooth vertex normal, butnormal
is not used anywhere else.Expand source code
class CSGVertex(object): """ Class CSGVertex Represents a vertex of a polygon. Use your own vertex class instead of this one to provide additional features like texture coordinates and vertex colors. Custom vertex classes need to provide a `pos` property and `clone()`, `flip()`, and `interpolate()` methods that behave analogous to the ones defined by `Vertex`. This class provides `normal` so convenience functions like `CSG.sphere()` can return a smooth vertex normal, but `normal` is not used anywhere else. """ def __init__(self, pos, normal=[0.0, 0.0, 0.0]): self.pos = CSGVector(pos[0], pos[1], pos[2]) self.normal = CSGVector(normal[0], normal[1], normal[2]) def clone(self): return CSGVertex(self.pos.clone(), self.normal.clone()) def flip(self): """ Invert all orientation-specific data (e.g. vertex normal). Called when the orientation of a polygon is flipped. """ self.normal = self.normal.negated() def interpolate(self, other, t): """ Create a new vertex between this vertex and `other` by linearly interpolating all properties using a parameter of `t`. Subclasses should override this to interpolate additional properties. """ return CSGVertex(self.pos.lerp(other.pos, t), self.normal.lerp(other.normal, t))
Methods
def clone(self)
-
Expand source code
def clone(self): return CSGVertex(self.pos.clone(), self.normal.clone())
def flip(self)
-
Invert all orientation-specific data (e.g. vertex normal). Called when the orientation of a polygon is flipped.
Expand source code
def flip(self): """ Invert all orientation-specific data (e.g. vertex normal). Called when the orientation of a polygon is flipped. """ self.normal = self.normal.negated()
def interpolate(self, other, t)
-
Create a new vertex between this vertex and
other
by linearly interpolating all properties using a parameter oft
. Subclasses should override this to interpolate additional properties.Expand source code
def interpolate(self, other, t): """ Create a new vertex between this vertex and `other` by linearly interpolating all properties using a parameter of `t`. Subclasses should override this to interpolate additional properties. """ return CSGVertex(self.pos.lerp(other.pos, t), self.normal.lerp(other.normal, t))