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)