Source code for sisl.shape.base

# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at https://mozilla.org/MPL/2.0/.
from __future__ import annotations

from math import sqrt as msqrt
from typing import Sequence

import numpy as np

import sisl._array as _a
from sisl._dispatch_class import _Dispatchs
from sisl._dispatcher import AbstractDispatch, ClassDispatcher
from sisl._internal import set_module
from sisl.messages import deprecation
from sisl.utils.mathematics import fnorm

__all__ = [
    "Shape",
    "PureShape",
    "NullShape",
    "ShapeToDispatch",
    "CompositeShape",
    "OrShape",
    "XOrShape",
    "AndShape",
    "SubShape",
]


@set_module("sisl.shape")
class Shape(
    _Dispatchs,
    dispatchs=[ClassDispatcher("to", type_dispatcher=None, obj_getattr="error")],
    when_subclassing="copy",
):
    """Baseclass for all shapes. Logical operations are implemented on this class.

    **This class must be sub classed.**

    Also all the required methods are predefined although they issue an error if they are
    not implemented in the sub-classed class.

    There are a few routines that are necessary when implementing
    an inherited class:

    `center`
      return the geometric center of the shape.

    `within`
      Returns a boolean array which defines whether a coordinate is within
      or outside the shape.

    `within_index`
      Equivalent to `within`, however only the indices of those within are returned.

    `copy`
      Create a new identical shape.

    The minimal requirement a shape can have are the above attributes.

    Subclassed shapes may have additional methods by which they are defined.

    Any `Shape` may be used to construct other shapes by applying set operations.
    Currently implemented binary operators are:

    `__or__`/`__add__` : set union, either `|` or `+` operator (not `or`)

    `__and__` : set intersection, `&` operator (not `and`)

    `__sub__` : set complement, `-` operator

    `__xor__` : set disjunctive union, `^` operator

    Parameters
    ----------
    center : (3,)
       the center of the shape
    """

    __slots__ = ("_center",)

[docs] def __init__(self, center=(0, 0, 0)): if center is None: center = (0, 0, 0) center = _a.asarrayd(center).flatten() if center.size == 0: center = _a.asarrayd([0, 0, 0]) elif center.size == 1: center = np.repeat(center, 3).flatten() if center.size != 3: raise ValueError( f"{self.__class__.__name__} requires setting a center to have 1/3 values." ) self._center = center
[docs] def copy(self): """Create a new copy of this object.""" return self.__class__(self.center)
@property def center(self): """The geometric center of the shape""" return self._center @center.setter def center(self, center): """Set the geometric center of the shape""" self._center[:] = center
[docs] def scale(self, scale): """Return a new Shape with scaled size""" raise NotImplementedError( f"{self.__class__.__name__}.scale has not been implemented" )
[docs] def translate(self, xyz: Sequence[float]): """Translate the center of the shape by `xyz`.""" copy = self.copy() copy.center = copy.center + xyz return copy
[docs] @deprecation( "toSphere is deprecated, use shape.to.Sphere(...) instead.", "0.15", "0.16" ) def toSphere(self, *args, **kwargs): """Create a sphere which is surely encompassing the *full* shape""" raise NotImplementedError( f"{self.__class__.__name__}.toSphere has not been implemented" )
[docs] @deprecation( "toEllipsoid is deprecated, use shape.to.Ellipsoid(...) instead.", "0.15", "0.16", ) def toEllipsoid(self, *args, **kwargs): """Create an ellipsoid which is surely encompassing the *full* shape""" return self.to.Sphere().to.Ellipsoid(*args, **kwargs)
[docs] @deprecation( "toCuboid is deprecated, use shape.to.Cuboid(...) instead.", "0.15", "0.16" ) def toCuboid(self, *args, **kwargs): """Create a cuboid which is surely encompassing the *full* shape""" return self.to.Ellipsoid().to.Cuboid(*args, **kwargs)
[docs] def within(self, other, *args, **kwargs): """Return ``True`` if `other` is fully within `self` If `other` is an array, an array will be returned for each of these. Parameters ---------- other : array_like the array/object that is checked for containment *args : passed directly to `within_index` **kwargs : passed directly to `within_index` """ other = _a.asarrayd(other) ndim = other.ndim other.shape = (-1, 3) idx = self.within_index(other, *args, **kwargs) # Initialize a boolean array with all false within = np.zeros(len(other), dtype=bool) within[idx] = True if ndim == 1 and other.size == 3: return within[0] return within
[docs] def within_index(self, other, *args, **kwargs): """Return indices of the elements of `other` that are within the shape""" raise NotImplementedError( f"{self.__class__.__name__}.within_index has not been implemented" )
def __contains__(self, other): """Checks whether all of `other` is within the shape""" return np.all(self.within(other)) def __str__(self): c = self.center return f"{self.__class__.__name__}{{c({c[0]} {c[1]} {c[2]})}}" # Implement logical operators to enable composition of sets def __and__(self, other): return AndShape(self, other) def __or__(self, other): return OrShape(self, other) def __add__(self, other): return OrShape(self, other) def __sub__(self, other): return SubShape(self, other) def __xor__(self, other): return XOrShape(self, other) to_dispatch = Shape.to # Add dispatcher systems class ShapeToDispatch(AbstractDispatch): """Base dispatcher from class passing from a Shape class""" class ToEllipsoidDispatch(ShapeToDispatch): def dispatch(self, *args, **kwargs): shape = self._get_object() return shape.to.Sphere(*args, **kwargs).to.Ellipsoid() to_dispatch.register("Ellipsoid", ToEllipsoidDispatch) class ToCuboidDispatch(ShapeToDispatch): def dispatch(self, *args, **kwargs): shape = self._get_object() return shape.to.Ellipsoid(*args, **kwargs).to.Cuboid() to_dispatch.register("Cuboid", ToCuboidDispatch) @set_module("sisl.shape") class CompositeShape(Shape): """A composite shape consisting of two shapes, an abstract class This should take 2 shapes as arguments. Parameters ---------- A : Shape the left hand side of the set operation B : Shape the right hand side of the set operation """ __slots__ = ("A", "B") def __init__(self, A, B): self.A = A.copy() self.B = B.copy() def __init_subclass__(cls, /, composite_name: str, **kwargs): super().__init_subclass__(**kwargs) cls.__slots__ = () cls.__str__ = _composite_name(composite_name) def copy(self): """Create a new copy of this object.""" return self.__class__(self.A, self.B) @property def center(self): """Average center of composite shapes""" return (self.A.center + self.B.center) * 0.5 @center.setter def center(self, center): """Set the geometric center of the shape""" self.A.center[:] = center self.B.center[:] = center @property def volume(self): """Volume of a composite shape is current undefined, so a negative number is returned (may change)""" # The volume for these set operators cannot easily be defined, so # we should rather not do anything about it. # TODO we could *estimate* the volume by doing # self.toSphere() # grid of density 0.01 # within_index # and calculate fractional volume # This is very inaccurate, but would probably be # good enough. return -1.0 @deprecation( "toSphere is deprecated, use shape.to.Sphere(...) instead.", "0.15", "0.16" ) def toSphere(self, *args, **kwargs): """Create a sphere which is surely encompassing the *full* shape""" return self.to.Sphere(*args, **kwargs) def scale(self, scale): """Return a new Shape with scaled center coords and shapes""" return self.__class__(self.A.scale(scale), self.B.scale(scale)) class ToSphereDispatch(ShapeToDispatch): def dispatch(self, center=None): """Create a sphere which is surely encompassing the *full* shape""" from .ellipsoid import Sphere shape = self._get_object() # Retrieve spheres A = shape.A.to.Sphere() Ar = A.radius Ac = A.center B = shape.B.to.Sphere() Br = B.radius Bc = B.center center_shape = (Ac + Bc) * 0.5 if center is None: center = center_shape A = Ar + fnorm(center_shape - Ac) B = Br + fnorm(center_shape - Bc) return Sphere(max(A, B), center) CompositeShape.to.register("Sphere", ToSphereDispatch) class ToEllipsoidDispatcher(ShapeToDispatch): def dispatch(self, *args, center=None, **kwargs): from .ellipsoid import Ellipsoid shape = self._get_object() return shape.to.Sphere(center=center).to.Ellipsoid() CompositeShape.to.register("Ellipsoid", ToEllipsoidDispatch) def _composite_name(sep): def _str(self): if isinstance(self.A, CompositeShape): A = "({})".format(str(self.A).replace("\n", "\n ")) else: A = "{}".format(str(self.A).replace("\n", "\n ")) if isinstance(self.B, CompositeShape): B = "({})".format(str(self.B).replace("\n", "\n ")) else: B = "{}".format(str(self.B).replace("\n", "\n ")) return f"{self.__class__.__name__}{{{A} {sep} {B}}}" return _str @set_module("sisl.shape") class OrShape(CompositeShape, composite_name="|"): """Boolean ``A | B`` shape""" def within_index(self, *args, **kwargs): A = self.A.within_index(*args, **kwargs) B = self.B.within_index(*args, **kwargs) return np.union1d(A, B) @set_module("sisl.shape") class XOrShape(CompositeShape, composite_name="^"): """Boolean ``A ^ B`` shape""" def within_index(self, *args, **kwargs): A = self.A.within_index(*args, **kwargs) B = self.B.within_index(*args, **kwargs) return np.setxor1d(A, B, assume_unique=True) @set_module("sisl.shape") class SubShape(CompositeShape, composite_name="-"): """Boolean ``A - B`` shape""" def within_index(self, *args, **kwargs): A = self.A.within_index(*args, **kwargs) B = self.B.within_index(*args, **kwargs) return np.setdiff1d(A, B, assume_unique=True) @set_module("sisl.shape") class AndShape(CompositeShape, composite_name="&"): """Boolean ``A & B`` shape""" @deprecation( "toSphere is deprecated, use shape.to.Sphere(...) instead.", "0.15", "0.16" ) def toSphere(self, *args, **kwargs): """Create a sphere which is surely encompassing the *full* shape""" return self.to.Sphere(*args, **kwargs) def within_index(self, *args, **kwargs): A = self.A.within_index(*args, **kwargs) B = self.B.within_index(*args, **kwargs) return np.intersect1d(A, B, assume_unique=True) class AndToSphereDispatch(ShapeToDispatch): def dispatch(self, center=None): """Create a sphere which is surely encompassing the *full* shape""" from .ellipsoid import Sphere shape = self._obj # Retrieve spheres A = shape.A.to.Sphere() Ar = A.radius Ac = A.center B = shape.B.to.Sphere() Br = B.radius Bc = B.center # Calculate the distance between the spheres dist = fnorm(Ac - Bc) # If one is fully enclosed in the other, we can simply neglect the other if dist + Ar <= Br: # A is fully enclosed in B (or they are the same) return A elif dist + Br <= Ar: # B is fully enclosed in A (or they are the same) return B elif dist <= (Ar + Br): # We can reduce the sphere drastically because only the overlapping region is important # i_r defines the intersection radius, search for Sphere-Sphere Intersection dx = (dist**2 - Br**2 + Ar**2) / (2 * dist) if dx > dist: # the intersection is placed after the radius of B # And in this case B is smaller (otherwise dx < 0) return B elif dx < 0: return A i_r = msqrt(4 * (dist * Ar) ** 2 - (dist**2 - Br**2 + Ar**2) ** 2) / ( 2 * dist ) # Now we simply need to find the dx point along the vector Bc - Ac # Then we can easily calculate the point from A if center is None: center = Bc - Ac center = Ac + center / fnorm(center) * dx A = i_r B = i_r else: # In this case there is actually no overlap. So perhaps we should # create an infinitisemal sphere such that no point will ever be found # Or we should return a new Shape which *always* return False for indices etc. if center is None: center = (Ac + Bc) * 0.5 return NullShape().to.Sphere(center=center) return Sphere(max(A, B), center) AndShape.to.register("Sphere", AndToSphereDispatch) @set_module("sisl.shape") class PureShape(Shape): """Extension of the `Shape` class for additional well defined shapes This shape should be used when subclassing shapes where the volume of the shape is *exactly* known. `volume` return the volume of the shape. """ __slots__ = () @property def volume(self): raise NotImplementedError( f"{self.__class__.__name__}.volume has not been implemented" ) def expand(self, _): """Expand the shape by a constant value""" raise NotImplementedError( f"{self.__class__.__name__}.expand has not been implemented" ) @set_module("sisl.shape") class NullShape(PureShape, dispatchs=[("to", "copy")]): """A unique shape which has no well-defined spatial volume or center This special shape is used when composite shapes turns out to have a null space. The center will be equivalent to the maximum floating point value divided by 100. Initialization of the NullShape takes no (or any) arguments. Since it has no volume of point in space, none of the arguments has any meaning. """ __slots__ = ()
[docs] def __init__(self, *args, **kwargs): """Initialize a null-shape""" M = np.finfo(np.float64).max / 100 self._center = np.array([M, M, M], np.float64)
[docs] def within_index(self, other, *args, **kwargs): """Always returns a zero length array""" return np.empty(0, dtype=np.int32)
[docs] @deprecation( "toEllipsoid is deprecated, use shape.to.Ellipsoid(...) instead.", "0.15", "0.16", ) def toEllipsoid(self, *args, **kwargs): """Return an ellipsoid with radius of size 1e-64""" return self.to.Ellipsoid(*args, **kwargs)
[docs] @deprecation( "toSphere is deprecated, use shape.to.Sphere(...) instead.", "0.15", "0.16" ) def toSphere(self, *args, **kwargs): """Return a sphere with radius of size 1e-64""" return self.to.Sphere(*args, **kwargs)
[docs] @deprecation( "toCuboid is deprecated, use shape.to.Cuboid(...) instead.", "0.15", "0.16" ) def toCuboid(self, *args, **kwargs): """Return a cuboid with side-lengths 1e-64""" return self.to.Cuboid(*args, **kwargs)
@property def volume(self): """The volume of a null shape is exactly 0.""" return 0.0 to_dispatch = NullShape.to class NullToSphere(ShapeToDispatch): def dispatch(self, *args, center=None, **kwargs): from .ellipsoid import Sphere shape = self._get_object() if center is None: center = shape.center.copy() return Sphere(1.0e-64, center=center) to_dispatch.register("Sphere", NullToSphere) class NullToEllipsoid(ShapeToDispatch): def dispatch(self, *args, center=None, **kwargs): from .ellipsoid import Ellipsoid shape = self._get_object() if center is None: center = shape.center.copy() return Ellipsoid(1.0e-64, center=center) to_dispatch.register("Ellipsoid", NullToEllipsoid) class NullToCuboid(ShapeToDispatch): def dispatch(self, *args, center=None, origin=None, **kwargs): from .prism4 import Cuboid shape = self._get_object() if center is None and origin is None: center = shape.center.copy() return Cuboid(1.0e-64, center=center, origin=origin) to_dispatch.register("Cuboid", NullToCuboid) del to_dispatch