Module sverchok.utils.mesh_spatial

Expand source code
# ##### BEGIN GPL LICENSE BLOCK #####
#
#  This program is free software; you can redistribute it and/or
#  modify it under the terms of the GNU General Public License
#  as published by the Free Software Foundation; either version 2
#  of the License, or (at your option) any later version.
#
#  This program is distributed in the hope that it will be useful,
#  but WITHOUT ANY WARRANTY; without even the implied warranty of
#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#  GNU General Public License for more details.
#
#  You should have received a copy of the GNU General Public License
#  along with this program; if not, write to the Free Software Foundation,
#  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# ##### END GPL LICENSE BLOCK #####

from mathutils import Vector
from mathutils.geometry import intersect_point_line
try:
    from mathutils.geometry import delaunay_2d_cdt
except ImportError:
    pass
from mathutils.bvhtree import BVHTree
import bmesh

from sverchok.data_structure import rotate_list
from sverchok.utils.geom import linear_approximation
from sverchok.utils.sv_bmesh_utils import bmesh_from_pydata, pydata_from_bmesh

def is_on_edge(pt, v1, v2, epsilon=1e-6):
    nearest, percentage = intersect_point_line(pt, v1, v2)
    distance = (Vector(pt) - nearest).length
    return (0.0 <= percentage <= 1.0) and (distance < epsilon)

def is_on_face_edge(pt, face_verts, epsilon=1e-6):
    for v1, v2 in zip(face_verts, rotate_list(face_verts)):
        if is_on_edge(pt, v1, v2, epsilon):
            return True
    return False

def single_face_delaunay(face_verts, add_verts, epsilon=1e-6, exclude_boundary=True):
    n = len(face_verts)
    face = list(range(n))
    edges = [(i,i+1) for i in range(n-1)] + [(n-1, 0)]
    plane = linear_approximation(face_verts).most_similar_plane()
    face_verts_2d = [plane.point_uv_projection(v) for v in face_verts]
    if exclude_boundary:
        add_verts = [v for v in add_verts if not is_on_face_edge(v, face_verts, epsilon)]
    add_verts_2d = [plane.point_uv_projection(v) for v in add_verts]
    TRIANGLES = 1
    res = delaunay_2d_cdt(face_verts_2d + add_verts_2d, edges, [face], TRIANGLES, epsilon)
    new_verts_2d = res[0]
    new_edges = res[1]
    new_faces = res[2]
    new_add_verts = [tuple(plane.evaluate(p[0], p[1], normalize=True)) for p in new_verts_2d[n:]]
    return face_verts + new_add_verts, new_edges, new_faces

def mesh_insert_verts(verts, faces, add_verts_by_face, epsilon=1e-6, exclude_boundary=True, recalc_normals=True, preserve_shape=False):
    if preserve_shape:
        bvh = BVHTree.FromPolygons(verts, faces)
    bm = bmesh_from_pydata(verts, [], [])
    for face_idx, face in enumerate(faces):
        n = len(face)
        add_verts = add_verts_by_face.get(face_idx)

        if not add_verts:
            bm_verts = [bm.verts[idx] for idx in face]
            bm.faces.new(bm_verts)
            bm.faces.index_update()
            continue

        done_verts = dict((i, bm.verts[face[i]]) for i in range(n))
        face_verts = [verts[i] for i in face]
        new_face_verts, edges, new_faces = single_face_delaunay(face_verts, add_verts, epsilon, exclude_boundary)
        if preserve_shape:
            new_face_verts = find_nearest_points(bvh, new_face_verts)
        for new_face in new_faces:
            bm_verts = []
            for i in new_face:
                bm_vert = done_verts.get(i)
                if not bm_vert:
                    bm_vert = bm.verts.new(new_face_verts[i])
                    bm.verts.index_update()
                    bm.verts.ensure_lookup_table()
                    done_verts[i] = bm_vert
                bm_verts.append(bm_vert)
                
            bm.faces.new(bm_verts)
            bm.faces.index_update()

    if recalc_normals:
        bmesh.ops.recalc_face_normals(bm, faces=bm.faces)

    verts, edges, faces = pydata_from_bmesh(bm)
    bm.free()
    return verts, edges, faces

def find_nearest_points(bvh, verts):
    locs = []
    for vert in verts:
        loc, normal, idx, distance = bvh.find_nearest(vert)
        locs.append(tuple(loc))
    return locs

def find_nearest_idxs(verts, faces, add_verts):
    bvh = BVHTree.FromPolygons(verts, faces)
    idxs = []
    for vert in add_verts:
        loc, normal, idx, distance = bvh.find_nearest(vert)
        idxs.append(idx)
    return idxs

Functions

def find_nearest_idxs(verts, faces, add_verts)
Expand source code
def find_nearest_idxs(verts, faces, add_verts):
    bvh = BVHTree.FromPolygons(verts, faces)
    idxs = []
    for vert in add_verts:
        loc, normal, idx, distance = bvh.find_nearest(vert)
        idxs.append(idx)
    return idxs
def find_nearest_points(bvh, verts)
Expand source code
def find_nearest_points(bvh, verts):
    locs = []
    for vert in verts:
        loc, normal, idx, distance = bvh.find_nearest(vert)
        locs.append(tuple(loc))
    return locs
def is_on_edge(pt, v1, v2, epsilon=1e-06)
Expand source code
def is_on_edge(pt, v1, v2, epsilon=1e-6):
    nearest, percentage = intersect_point_line(pt, v1, v2)
    distance = (Vector(pt) - nearest).length
    return (0.0 <= percentage <= 1.0) and (distance < epsilon)
def is_on_face_edge(pt, face_verts, epsilon=1e-06)
Expand source code
def is_on_face_edge(pt, face_verts, epsilon=1e-6):
    for v1, v2 in zip(face_verts, rotate_list(face_verts)):
        if is_on_edge(pt, v1, v2, epsilon):
            return True
    return False
def mesh_insert_verts(verts, faces, add_verts_by_face, epsilon=1e-06, exclude_boundary=True, recalc_normals=True, preserve_shape=False)
Expand source code
def mesh_insert_verts(verts, faces, add_verts_by_face, epsilon=1e-6, exclude_boundary=True, recalc_normals=True, preserve_shape=False):
    if preserve_shape:
        bvh = BVHTree.FromPolygons(verts, faces)
    bm = bmesh_from_pydata(verts, [], [])
    for face_idx, face in enumerate(faces):
        n = len(face)
        add_verts = add_verts_by_face.get(face_idx)

        if not add_verts:
            bm_verts = [bm.verts[idx] for idx in face]
            bm.faces.new(bm_verts)
            bm.faces.index_update()
            continue

        done_verts = dict((i, bm.verts[face[i]]) for i in range(n))
        face_verts = [verts[i] for i in face]
        new_face_verts, edges, new_faces = single_face_delaunay(face_verts, add_verts, epsilon, exclude_boundary)
        if preserve_shape:
            new_face_verts = find_nearest_points(bvh, new_face_verts)
        for new_face in new_faces:
            bm_verts = []
            for i in new_face:
                bm_vert = done_verts.get(i)
                if not bm_vert:
                    bm_vert = bm.verts.new(new_face_verts[i])
                    bm.verts.index_update()
                    bm.verts.ensure_lookup_table()
                    done_verts[i] = bm_vert
                bm_verts.append(bm_vert)
                
            bm.faces.new(bm_verts)
            bm.faces.index_update()

    if recalc_normals:
        bmesh.ops.recalc_face_normals(bm, faces=bm.faces)

    verts, edges, faces = pydata_from_bmesh(bm)
    bm.free()
    return verts, edges, faces
def single_face_delaunay(face_verts, add_verts, epsilon=1e-06, exclude_boundary=True)
Expand source code
def single_face_delaunay(face_verts, add_verts, epsilon=1e-6, exclude_boundary=True):
    n = len(face_verts)
    face = list(range(n))
    edges = [(i,i+1) for i in range(n-1)] + [(n-1, 0)]
    plane = linear_approximation(face_verts).most_similar_plane()
    face_verts_2d = [plane.point_uv_projection(v) for v in face_verts]
    if exclude_boundary:
        add_verts = [v for v in add_verts if not is_on_face_edge(v, face_verts, epsilon)]
    add_verts_2d = [plane.point_uv_projection(v) for v in add_verts]
    TRIANGLES = 1
    res = delaunay_2d_cdt(face_verts_2d + add_verts_2d, edges, [face], TRIANGLES, epsilon)
    new_verts_2d = res[0]
    new_edges = res[1]
    new_faces = res[2]
    new_add_verts = [tuple(plane.evaluate(p[0], p[1], normalize=True)) for p in new_verts_2d[n:]]
    return face_verts + new_add_verts, new_edges, new_faces