Module sverchok.utils.mesh_functions

Some benchmark

m = Matrix.Identity(4) v = Vector((1,0,0)) timeit('(m @ v)[:]', 'from main import m, v') <— 0.2413583000000017 timeit('tuple(m @ v)', 'from main import m, v') <— 0.5317709999999352

Expand source code
# This file is part of project Sverchok. It's copyrighted by the contributors
# recorded in the version control history of the file, available from
# its original location https://github.com/nortikin/sverchok/commit/master
#
# SPDX-License-Identifier: GPL3
# License-Filename: LICENSE


"""
Some benchmark

m = Matrix.Identity(4)
v = Vector((1,0,0))
timeit('(m @ v)[:]', 'from __main__ import m, v')  <--- 0.2413583000000017
timeit('tuple(m @ v)', 'from __main__ import m, v')  <--- 0.5317709999999352
"""

from functools import wraps
from itertools import cycle, count, tee
from typing import Iterator, Tuple, List, Union

import numpy as np

from mathutils import Matrix, Vector
from sverchok.utils.modules.matrix_utils import matrix_apply_np

Vertex = Tuple[float, float, float]
Edge = Tuple[int, int]
Polygon = List[int]
PyMesh = Tuple[List[Vertex], List[Edge], List[Polygon]]

NpVertexes = np.ndarray
NpMesh = Tuple[NpVertexes, List[Edge], List[Polygon]]

Mesh = Union[PyMesh, NpMesh]


def meshes_py(vertices: List[List[Vertex]],
              edges: List[List[Edge]] = None,
              polygons: List[List[Polygon]] = None) -> Iterator[PyMesh]:
    yield from zip(vertices, edges or cycle([None]), polygons or cycle([None]))


def meshes_np(vertices: List[NpVertexes],
              edges: List[List[Edge]] = None,
              polygons: List[List[Polygon]] = None) -> Iterator[NpMesh]:
    vertices = np.asarray(vertices, dtype=np.float32)
    yield from zip(vertices, edges or cycle([None]), polygons or cycle([None]))


def pass_mesh_type(gen_func):
    @wraps(gen_func)
    def wrapper(meshes: Iterator[Mesh], *args, **kwargs):
        meshes, _meshes = tee(meshes)
        vertices, edges, polygons = next(_meshes)
        mesh_type = 'NP' if isinstance(vertices, np.ndarray) else 'PY'
        return gen_func(meshes, *args, _mesh_type=mesh_type, **kwargs)
    return wrapper


@pass_mesh_type
def apply_matrix(meshes: Iterator[Mesh], matrices: Iterator[Matrix], *, _mesh_type) -> Iterator[Mesh]:
    """It will generate new vertices with given matrix applied"""
    implementation = matrix_apply_np if _mesh_type == 'NP' else apply_matrix_to_vertices_py
    for (vertices, edges, polygons), matrix in zip(meshes, matrices):
        # several matrices can be applied to a mesh
        # in this case each matrix will populate geometry inside object
        sub_vertices = implementation(vertices, matrix)
        yield sub_vertices, edges, polygons


@pass_mesh_type
def apply_matrices(meshes: Iterator[Mesh], matrices: Iterator[List[Matrix]], *, _mesh_type) -> Iterator[Mesh]:
    """It will generate new vertices with given matrix applied"""
    implementation = matrix_apply_np if _mesh_type == 'NP' else apply_matrix_to_vertices_py
    for (vertices, edges, polygons), _matrices in zip(meshes, matrices):
        # several matrices can be applied to a mesh
        # in this case each matrix will populate geometry inside object
        sub_vertices = []
        sub_edges = [edges] * len(_matrices) if edges else None
        sub_polygons = [polygons] * len(_matrices) if polygons else None
        for matrix in _matrices:
            sub_vertices.append(implementation(vertices, matrix))

        yield from join_meshes(meshes_py(sub_vertices, sub_edges, sub_polygons))

def has_element(pol_edge):
    if pol_edge is None:
        return False
    if len(pol_edge) > 0 and len(pol_edge[0]) > 0:
        return True
    return False


@pass_mesh_type
def join_meshes(meshes: Iterator[PyMesh], *, _mesh_type) -> Iterator[PyMesh]:
    _vertices = []
    joined_edges = []
    joined_polygons = []
    vertexes_number = 0
    for vertices, edges, polygons in meshes:
        if has_element(edges):
            if isinstance(edges, np.ndarray):
                joined_edges.extend((edges + vertexes_number).tolist())
            else:
                joined_edges.extend([(e[0] + vertexes_number, e[1] + vertexes_number) for e in edges])
        if has_element(polygons):
            if isinstance(edges, np.ndarray):
                joined_polygons.extend((polygons + vertexes_number).tolist())
            else:
                joined_polygons.extend([[i + vertexes_number for i in p] for p in polygons])
        vertexes_number += len(vertices)
        _vertices.append(vertices)
    implementation = np.concatenate if _mesh_type == 'NP' else lambda vs: [v for _vs in vs for v in _vs]
    joined_vertices = implementation(_vertices)
    yield joined_vertices, joined_edges, joined_polygons


def to_elements(meshes: Iterator[PyMesh]) -> Tuple[List[List[Vertex]],
                                                   List[List[Edge]],
                                                   List[List[Polygon]]]:
    vertices, edges, polygons = zip(*meshes)
    edges = edges if edges[0] else []
    polygons = polygons if polygons[0] else []
    return vertices, edges, polygons


def repeat_meshes(meshes: Iterator[Mesh], number: int = -1) -> Iterator[Mesh]:
    counter = count()
    last = None
    while next(counter) != number:
        try:
            last = next(meshes)
        except StopIteration:
            pass
        yield last


def apply_matrix_to_vertices_py(vertices: List[Vertex], matrix: Matrix) -> List[Vertex]:
    return [(matrix @ Vector(v)).to_tuple() for v in vertices]

def mesh_join(vertices: List[List[Vertex]],
              edges: List[List[Edge]],
              polygons: List[List[Polygon]]) -> Tuple[List[List[Vertex]],
                                                      List[List[Edge]],
                                                      List[List[Polygon]]]:
    is_py_input = isinstance(vertices[0], (list, tuple))
    meshes = (meshes_py if is_py_input else meshes_np)(vertices, edges, polygons)
    meshes = join_meshes(meshes)
    out_vertices, out_edges, out_polygons = to_elements(meshes)

    return out_vertices, out_edges, out_polygons

Functions

def apply_matrices(meshes: Iterator[Union[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]], Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]], matrices: Iterator[List[Matrix]], *, _mesh_type) ‑> Iterator[Union[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]], Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]]

It will generate new vertices with given matrix applied

Expand source code
@pass_mesh_type
def apply_matrices(meshes: Iterator[Mesh], matrices: Iterator[List[Matrix]], *, _mesh_type) -> Iterator[Mesh]:
    """It will generate new vertices with given matrix applied"""
    implementation = matrix_apply_np if _mesh_type == 'NP' else apply_matrix_to_vertices_py
    for (vertices, edges, polygons), _matrices in zip(meshes, matrices):
        # several matrices can be applied to a mesh
        # in this case each matrix will populate geometry inside object
        sub_vertices = []
        sub_edges = [edges] * len(_matrices) if edges else None
        sub_polygons = [polygons] * len(_matrices) if polygons else None
        for matrix in _matrices:
            sub_vertices.append(implementation(vertices, matrix))

        yield from join_meshes(meshes_py(sub_vertices, sub_edges, sub_polygons))
def apply_matrix(meshes: Iterator[Union[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]], Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]], matrices: Iterator[Matrix], *, _mesh_type) ‑> Iterator[Union[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]], Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]]

It will generate new vertices with given matrix applied

Expand source code
@pass_mesh_type
def apply_matrix(meshes: Iterator[Mesh], matrices: Iterator[Matrix], *, _mesh_type) -> Iterator[Mesh]:
    """It will generate new vertices with given matrix applied"""
    implementation = matrix_apply_np if _mesh_type == 'NP' else apply_matrix_to_vertices_py
    for (vertices, edges, polygons), matrix in zip(meshes, matrices):
        # several matrices can be applied to a mesh
        # in this case each matrix will populate geometry inside object
        sub_vertices = implementation(vertices, matrix)
        yield sub_vertices, edges, polygons
def apply_matrix_to_vertices_py(vertices: List[Tuple[float, float, float]], matrix: Matrix) ‑> List[Tuple[float, float, float]]
Expand source code
def apply_matrix_to_vertices_py(vertices: List[Vertex], matrix: Matrix) -> List[Vertex]:
    return [(matrix @ Vector(v)).to_tuple() for v in vertices]
def has_element(pol_edge)
Expand source code
def has_element(pol_edge):
    if pol_edge is None:
        return False
    if len(pol_edge) > 0 and len(pol_edge[0]) > 0:
        return True
    return False
def join_meshes(meshes: Iterator[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]]], *, _mesh_type) ‑> Iterator[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]]]
Expand source code
@pass_mesh_type
def join_meshes(meshes: Iterator[PyMesh], *, _mesh_type) -> Iterator[PyMesh]:
    _vertices = []
    joined_edges = []
    joined_polygons = []
    vertexes_number = 0
    for vertices, edges, polygons in meshes:
        if has_element(edges):
            if isinstance(edges, np.ndarray):
                joined_edges.extend((edges + vertexes_number).tolist())
            else:
                joined_edges.extend([(e[0] + vertexes_number, e[1] + vertexes_number) for e in edges])
        if has_element(polygons):
            if isinstance(edges, np.ndarray):
                joined_polygons.extend((polygons + vertexes_number).tolist())
            else:
                joined_polygons.extend([[i + vertexes_number for i in p] for p in polygons])
        vertexes_number += len(vertices)
        _vertices.append(vertices)
    implementation = np.concatenate if _mesh_type == 'NP' else lambda vs: [v for _vs in vs for v in _vs]
    joined_vertices = implementation(_vertices)
    yield joined_vertices, joined_edges, joined_polygons
def mesh_join(vertices: List[List[Tuple[float, float, float]]], edges: List[List[Tuple[int, int]]], polygons: List[List[List[int]]]) ‑> Tuple[List[List[Tuple[float, float, float]]], List[List[Tuple[int, int]]], List[List[List[int]]]]
Expand source code
def mesh_join(vertices: List[List[Vertex]],
              edges: List[List[Edge]],
              polygons: List[List[Polygon]]) -> Tuple[List[List[Vertex]],
                                                      List[List[Edge]],
                                                      List[List[Polygon]]]:
    is_py_input = isinstance(vertices[0], (list, tuple))
    meshes = (meshes_py if is_py_input else meshes_np)(vertices, edges, polygons)
    meshes = join_meshes(meshes)
    out_vertices, out_edges, out_polygons = to_elements(meshes)

    return out_vertices, out_edges, out_polygons
def meshes_np(vertices: List[numpy.ndarray], edges: List[List[Tuple[int, int]]] = None, polygons: List[List[List[int]]] = None) ‑> Iterator[Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]
Expand source code
def meshes_np(vertices: List[NpVertexes],
              edges: List[List[Edge]] = None,
              polygons: List[List[Polygon]] = None) -> Iterator[NpMesh]:
    vertices = np.asarray(vertices, dtype=np.float32)
    yield from zip(vertices, edges or cycle([None]), polygons or cycle([None]))
def meshes_py(vertices: List[List[Tuple[float, float, float]]], edges: List[List[Tuple[int, int]]] = None, polygons: List[List[List[int]]] = None) ‑> Iterator[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]]]
Expand source code
def meshes_py(vertices: List[List[Vertex]],
              edges: List[List[Edge]] = None,
              polygons: List[List[Polygon]] = None) -> Iterator[PyMesh]:
    yield from zip(vertices, edges or cycle([None]), polygons or cycle([None]))
def pass_mesh_type(gen_func)
Expand source code
def pass_mesh_type(gen_func):
    @wraps(gen_func)
    def wrapper(meshes: Iterator[Mesh], *args, **kwargs):
        meshes, _meshes = tee(meshes)
        vertices, edges, polygons = next(_meshes)
        mesh_type = 'NP' if isinstance(vertices, np.ndarray) else 'PY'
        return gen_func(meshes, *args, _mesh_type=mesh_type, **kwargs)
    return wrapper
def repeat_meshes(meshes: Iterator[Union[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]], Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]], number: int = -1) ‑> Iterator[Union[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]], Tuple[numpy.ndarray, List[Tuple[int, int]], List[List[int]]]]]
Expand source code
def repeat_meshes(meshes: Iterator[Mesh], number: int = -1) -> Iterator[Mesh]:
    counter = count()
    last = None
    while next(counter) != number:
        try:
            last = next(meshes)
        except StopIteration:
            pass
        yield last
def to_elements(meshes: Iterator[Tuple[List[Tuple[float, float, float]], List[Tuple[int, int]], List[List[int]]]]) ‑> Tuple[List[List[Tuple[float, float, float]]], List[List[Tuple[int, int]]], List[List[List[int]]]]
Expand source code
def to_elements(meshes: Iterator[PyMesh]) -> Tuple[List[List[Vertex]],
                                                   List[List[Edge]],
                                                   List[List[Polygon]]]:
    vertices, edges, polygons = zip(*meshes)
    edges = edges if edges[0] else []
    polygons = polygons if polygons[0] else []
    return vertices, edges, polygons