Source code for aquaduct.geom.convexhull
# -*- coding: utf-8 -*-
# Aqua-Duct, a tool facilitating analysis of the flow of solvent molecules in molecular dynamic simulations
# Copyright (C) 2016-2018  Tomasz Magdziarz, Alicja Płuciennik, Michał Stolarczyk <info@aquaduct.pl>
# Copyright (C) 2019  Tomasz Magdziarz <info@aquaduct.pl>
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
from aquaduct import logger
import numpy as np
# from scipy.spatial import ConvexHull as SciPyConvexHull
from scipy.spatial import ConvexHull
from aquaduct.utils.helpers import uniqify
from aquaduct.geom.traces import vector_change_len
################################################################################
# SciPy ConvexHull improvements
[docs]class SciPyConvexHull(ConvexHull):
    def __init__(self, points, inflate=None, *args, **kwargs):
        super(SciPyConvexHull, self).__init__(points, *args, **kwargs)
        # at this stage we have chull
        if inflate:
            inflate = float(inflate)
            center = np.mean(self.vertices_points, 0)
            # redo chull but inflate its vertices
            new_points = (vector_change_len(p - center, inflate) + center for p in points)
            super(SciPyConvexHull, self).__init__(list(new_points), *args, **kwargs)
    @property
    def vertices_ids(self):
        return self.vertices.tolist()
    @property
    def vertices_points(self):
        return self.points[self.vertices]
    @property
    def facets(self):
        for simp in self.simplices:
            yield np.vstack((self.points[simp], self.points[simp][0]))
    @property
    @uniqify
    def edges(self):
        for simp in self.simplices:
            n = len(simp)
            for nr, point in enumerate(simp):
                if nr < n - 1:
                    yield point, simp[nr + 1]
                else:
                    yield point, simp[0]
    @property
    def simplices_vertices(self):
        for simp in self.simplices:
            yield [self.vertices_ids.index(s) for s in simp]
    def point_within(self, point):
        # checks if point is within convexhull
        # build new convex hull with this point and vertices
        new_hull = SciPyConvexHull(np.vstack((point, self.vertices_points)))
        # return 0 not in _unfold_vertices_list(new_hull.vertices)
        return 0 not in new_hull.vertices_ids 
################################################################################
# primitives to check if point(s) is within chull
def is_point_within_convexhull(point_chull):
    # This is helper function to check if point is within convex hull.
    # point_chull is a tuple where first element holds a point and the second is a ConvexHull object.
    return point_chull[-1].point_within(point_chull[0])
# many points - new solution but soon will be deprecated
def are_points_within_convexhull(points, chull):
    vertices_points = chull.points[chull.vertices]
    promise = ((SciPyConvexHull(np.vstack((point, vertices_points))).vertices[0] != 0) for point in points)
    return np.fromiter(promise, bool)