# Wulfric - Crystal, Lattice, Atoms, K-path.
# Copyright (C) 2023-2024 Andrey Rybakov
#
# e-mail: anry@uv.es, web: adrybakov.com
#
# 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 <https://www.gnu.org/licenses/>.
from math import cos, sqrt
import numpy as np
from wulfric.constants import TODEGREES, TORADIANS
from wulfric.numerical import ABS_TOL, ABS_TOL_ANGLE, compare_numerically
__all__ = ["volume", "angle", "parallelepiped_check", "absolute_to_relative"]
[docs]
def volume(*args):
r"""
Computes volume.
Three type of arguments are expected:
* One argument.
Matrix ``cell``.
Volume is computed as:
.. math::
V = \boldsymbol{v_1} \cdot (\boldsymbol{v_2} \times \boldsymbol{v_3})
* Three arguments.
Vectors ``\boldsymbol{v_1}``, ``\boldsymbol{v_2}``, ``\boldsymbol{v_3}``.
Volume is computed as:
.. math::
V = \boldsymbol{v_1} \cdot (\boldsymbol{v_2} \times \boldsymbol{v_3})
* Six arguments.
Parameters ``a``, ``b``, ``c``, ``alpha``, ``beta``, ``gamma``.
Volume is computed as:
.. math::
V = abc\sqrt{1+2\cos(\alpha)\cos(\beta)\cos(\gamma)-\cos^2(\alpha)-\cos^2(\beta)-\cos^2(\gamma)}
Parameters
----------
v1 : (3,) |array-like|_
First vector.
v2 : (3,) |array-like|_
Second vector.
v3 : (3,) |array-like|_
Third vector.
cell : (3,3) |array-like|_
Cell matrix, rows are interpreted as vectors.
a : float, default 1
Length of the :math:`v_1` vector.
b : float, default 1
Length of the :math:`v_2` vector.
c : float, default 1
Length of the :math:`v_3` vector.
alpha : float, default 90
Angle between vectors :math:`v_2` and :math:`v_3`. In degrees.
beta : float, default 90
Angle between vectors :math:`v_1` and :math:`v_3`. In degrees.
gamma : float, default 90
Angle between vectors :math:`v_1` and :math:`v_2`. In degrees.
Returns
-------
volume : float
Volume of corresponding region in space.
"""
if len(args) == 1:
cell = np.array(args[0])
elif len(args) == 3:
cell = np.array(args)
elif len(args) == 6:
a, b, c, alpha, beta, gamma = args
alpha = alpha * TORADIANS
beta = beta * TORADIANS
gamma = gamma * TORADIANS
sq_root = (
1
+ 2 * cos(alpha) * cos(beta) * cos(gamma)
- cos(alpha) ** 2
- cos(beta) ** 2
- cos(gamma) ** 2
)
return a * b * c * sqrt(sq_root)
else:
raise ValueError(
"Unable to identify input. "
+ "Supported: one (3,3) array-like, or three (3,) array-like, or 6 floats."
)
return abs(np.linalg.det(cell))
[docs]
def angle(v1, v2, radians=False):
r"""
Angle between two vectors.
.. math::
\alpha
=
\dfrac{(\boldsymbol{v_1} \cdot \boldsymbol{v_2})}
{\vert\boldsymbol{v_1}\vert\cdot\vert\boldsymbol{v_2}\vert}
Parameters
----------
v1 : (3,) |array-like|_
First vector.
v2 : (3,) |array-like|_
Second vector.
radians : bool, default False
Whether to return value in radians.
Returns
-------
angle: float
Angle in degrees or radians (see ``radians``).
Raises
------
ValueError
If one of the vectors is zero vector (or both). Norm is compared against
:numpy:`finfo`\ (float).eps.
"""
# Normalize vectors
v1_norm = np.linalg.norm(v1)
v2_norm = np.linalg.norm(v2)
if abs(v1_norm) <= np.finfo(float).eps or abs(v2_norm) <= np.finfo(float).eps:
raise ValueError("Angle is ill defined (zero vector).")
v1 = np.array(v1) / v1_norm
v2 = np.array(v2) / v2_norm
alpha = np.arccos(np.clip(np.dot(v1, v2), -1.0, 1.0))
if radians:
return alpha
return alpha * TODEGREES
[docs]
def parallelepiped_check(a, b, c, alpha, beta, gamma, raise_error=False):
r"""
Check if parallelepiped is valid.
The following checks are performed:
* :math:`a > 0`
* :math:`b > 0`
* :math:`c > 0`
* :math:`0 < \alpha < 180`
* :math:`0 < \beta < 180`
* :math:`0 < \gamma < 180`
* :math:`\gamma < \alpha + \beta < 360 - \gamma`
* :math:`\beta < \alpha + \gamma < 360 - \beta`
* :math:`\alpha < \beta + \gamma < 360 - \alpha`
Parameters
----------
a : float
Length of the :math:`\boldsymbol{v_1}` vector.
b : float
Length of the :math:`\boldsymbol{v_2}` vector.
c : float
Length of the :math:`\boldsymbol{v_3}` vector.
alpha : float
Angle between vectors :math:`\boldsymbol{v_2}` and :math:`\boldsymbol{v_3}`. In degrees.
beta : float
Angle between vectors :math:`\boldsymbol{v_1}` and :math:`\boldsymbol{v_3}`. In degrees.
gamma : float
Angle between vectors :math:`\boldsymbol{v_1}` and :math:`\boldsymbol{v_2}`. In degrees.
raise_error : bool, default False
Whether to raise error if parameters can not form a parallelepiped.
Returns
-------
result: bool
Whether the parameters could from a parallelepiped.
Raises
------
ValueError
If parameters could not form a parallelepiped.
Only raised if ``raise_error`` is ``True`` (it is ``False`` by default).
"""
result = (
compare_numerically(a, ">", 0.0, ABS_TOL)
and compare_numerically(b, ">", 0.0, ABS_TOL)
and compare_numerically(c, ">", 0.0, ABS_TOL)
and compare_numerically(alpha, "<", 180.0, ABS_TOL_ANGLE)
and compare_numerically(beta, "<", 180.0, ABS_TOL_ANGLE)
and compare_numerically(gamma, "<", 180.0, ABS_TOL_ANGLE)
and compare_numerically(alpha, ">", 0.0, ABS_TOL_ANGLE)
and compare_numerically(beta, ">", 0.0, ABS_TOL_ANGLE)
and compare_numerically(gamma, ">", 0.0, ABS_TOL_ANGLE)
and compare_numerically(gamma, "<", alpha + beta, ABS_TOL_ANGLE)
and compare_numerically(alpha + beta, "<", 360.0 - gamma, ABS_TOL_ANGLE)
and compare_numerically(beta, "<", alpha + gamma, ABS_TOL_ANGLE)
and compare_numerically(alpha + gamma, "<", 360.0 - beta, ABS_TOL_ANGLE)
and compare_numerically(alpha, "<", beta + gamma, ABS_TOL_ANGLE)
and compare_numerically(beta + gamma, "<", 360.0 - alpha, ABS_TOL_ANGLE)
)
if not result and raise_error:
message = "Parameters could not form a parallelepiped:\n"
message += f"a = {a}"
if not compare_numerically(a, ">", 0.0, ABS_TOL):
message += f" (a <= 0 with numerical tolerance: {ABS_TOL})"
message += "\n"
message += f"b = {b}"
if not compare_numerically(b, ">", 0.0, ABS_TOL):
message += f" (b <= 0 with numerical tolerance: {ABS_TOL})"
message += "\n"
message += f"c = {c}"
if not compare_numerically(c, ">", 0.0, ABS_TOL):
message += f" (c <= 0 with numerical tolerance: {ABS_TOL})"
message += "\n"
message += f"alpha = {alpha}\n"
if not compare_numerically(alpha, "<", 180.0, ABS_TOL_ANGLE):
message += f" (alpha >= 180 with numerical tolerance: {ABS_TOL_ANGLE})\n"
if not compare_numerically(alpha, ">", 0.0, ABS_TOL_ANGLE):
message += f" (alpha <= 0 with numerical tolerance: {ABS_TOL_ANGLE})\n"
message += f"beta = {beta}\n"
if not compare_numerically(beta, "<", 180.0, ABS_TOL_ANGLE):
message += f" (beta >= 180 with numerical tolerance: {ABS_TOL_ANGLE})\n"
if not compare_numerically(beta, ">", 0.0, ABS_TOL_ANGLE):
message += f" (beta <= 0 with numerical tolerance: {ABS_TOL_ANGLE})\n"
message += f"gamma = {gamma}\n"
if not compare_numerically(gamma, "<", 180.0, ABS_TOL_ANGLE):
message += f" (gamma >= 180 with numerical tolerance: {ABS_TOL_ANGLE})\n"
if not compare_numerically(gamma, ">", 0.0, ABS_TOL_ANGLE):
message += f" (gamma <= 0 with numerical tolerance: {ABS_TOL_ANGLE})\n"
if not compare_numerically(gamma, "<", alpha + beta, ABS_TOL_ANGLE):
message += f"Inequality gamma < alpha + beta is not satisfied with numerical tolerance: {ABS_TOL_ANGLE}\n"
if not compare_numerically(alpha + beta, "<", 360.0 - gamma, ABS_TOL_ANGLE):
message += f"Inequality alpha + beta < 360 - gamma is not satisfied with numerical tolerance: {ABS_TOL_ANGLE}\n"
if not compare_numerically(beta, "<", alpha + gamma, ABS_TOL_ANGLE):
message += f"Inequality beta < alpha + gamma is not satisfied with numerical tolerance: {ABS_TOL_ANGLE}\n"
if not compare_numerically(alpha + gamma, "<", 360.0 - beta, ABS_TOL_ANGLE):
message += f"Inequality alpha + gamma < 360 - beta is not satisfied with numerical tolerance: {ABS_TOL_ANGLE}\n"
if not compare_numerically(alpha, "<", beta + gamma, ABS_TOL_ANGLE):
message += f"Inequality alpha < beta + gamma is not satisfied with numerical tolerance: {ABS_TOL_ANGLE}\n"
if not compare_numerically(beta + gamma, "<", 360.0 - alpha, ABS_TOL_ANGLE):
message += f"Inequality beta + gamma < 360 - alpha is not satisfied with numerical tolerance: {ABS_TOL_ANGLE}\n"
raise ValueError(message)
return result
[docs]
def absolute_to_relative(vector, basis):
r"""
Compute relative coordinates of the vector with respect to the basis.
.. math::
\boldsymbol{v} = v^1 \boldsymbol{e_1} + v^2 \boldsymbol{e_2} + v^3 \boldsymbol{e_3}
We compute scalar products of the vector with the basis vectors:
.. math::
\begin{matrix}
\boldsymbol{v} \cdot \boldsymbol{e_1}
=
v^1\, \boldsymbol{e_1} \cdot \boldsymbol{e_1}
+
v^2\, \boldsymbol{e_2} \cdot \boldsymbol{e_1}
+
v^3\, \boldsymbol{e_3} \cdot \boldsymbol{e_1} \\
\boldsymbol{v} \cdot \boldsymbol{e_2}
=
v^1\, \boldsymbol{e_1} \cdot \boldsymbol{e_2}
+
v^2\, \boldsymbol{e_2} \cdot \boldsymbol{e_2}
+
v^3\, \boldsymbol{e_3} \cdot \boldsymbol{e_2} \\
\boldsymbol{v} \cdot \boldsymbol{e_3}
=
v^1\, \boldsymbol{e_1} \cdot \boldsymbol{e_3}
+
v^2\, \boldsymbol{e_2} \cdot \boldsymbol{e_3}
+
v^3\, \boldsymbol{e_3} \cdot \boldsymbol{e_3}
\end{matrix}
Which leads to the system of linear equations for :math:`v^1`, :math:`v^2`, :math:`v^3`.
Parameters
----------
vector : (3,) |array-like|_
Absolute coordinates of a vector.
basis : (3, 3) |array-like|_
Basis vectors. Rows are interpreted as vectors.
Columns are interpreted as coordinates.
Returns
-------
relative : (3,) :numpy:`ndarray`
Relative coordinates of the ``vector`` with respect to the ``basis``.
:math:`(v^1, v^2, v^3)`.
"""
# Three vectors of the basis
e1 = np.array(basis[0], dtype=float)
e2 = np.array(basis[1], dtype=float)
e3 = np.array(basis[2], dtype=float)
v = np.array(vector, dtype=float)
if (v == np.zeros(3)).all():
return np.zeros(3)
# Compose system of linear equations
B = np.array([np.dot(e1, v), np.dot(e2, v), np.dot(e3, v)])
A = np.array(
[
[np.dot(e1, e1), np.dot(e1, e2), np.dot(e1, e3)],
[np.dot(e2, e1), np.dot(e2, e2), np.dot(e2, e3)],
[np.dot(e3, e1), np.dot(e3, e2), np.dot(e3, e3)],
]
)
# Solve and return
return np.linalg.solve(A, B)