Source code for storylines.calc

# Copyright (C) 2016-2025 Jan Berges
# This program is free software under the terms of the BSD Zero Clause License.

"""Mathematical helpers."""

from __future__ import division

import math

[docs] def order_of_magnitude(x): """Calculate the decimal order of magnitude. Parameters ---------- x : float Number of which to calculate the decimal order of magnitude. Returns ------- int Order of magnitude of `x`. """ return int(math.floor(math.log10(abs(x)))) if x else 0
[docs] def power_of_ten(x): """Calculate the power of ten of the same order of magnitude. Parameters ---------- x : float Number of which to calculate the power of ten of the same order of magnitude. Returns ------- float Power of ten of the same order of magnitude as `x`. """ return 10 ** order_of_magnitude(x)
[docs] def xround(x, divisor=1): """Round to multiple of given number. Parameters ---------- x : float Number to round. divisor : float Number the result shall be a multiple of. Returns ------- float `x` rounded to the closest multiple of `divisor`. """ return divisor * round(x / divisor)
[docs] def xround_mantissa(x, divisor=1): """Round mantissa to multiple of given number. The mantissa is the part before the power of ten in scientific notation. Parameters ---------- x : float Number the mantissa of which to round. divisor : float Number the rounded mantissa shall be a multiple of. Returns ------- float `x` with the mantissa rounded to the closest multiple of `divisor`. """ return xround(x, divisor * power_of_ten(x))
[docs] def multiples(lower, upper, divisor=1): """Iterate over all integer multiples of given number on closed interval. Parameters ---------- lower, upper : float Bounds of closed interval. divisor : float Number the results shall be multiples of. Yields ------ float Multiple of `divisor` between `lower` and `upper`. """ for n in range(int(math.ceil(lower / divisor)), int(math.floor(upper / divisor)) + 1): yield divisor * n
[docs] def multiply(A, b): """Multiply vector by scalar. Parameters ---------- A : list of float Vector. b : float Scalar. Returns ------- list of float `A` multiplied by `b`. """ return [a * b for a in A]
[docs] def divide(A, b): """Divide vector by scalar. Parameters ---------- A : list of float Vector. b : float Scalar. Returns ------- list of float `A` divided by `b`. """ return [a / b for a in A]
[docs] def add(A, B): """Calculate sum of two vectors. Parameters ---------- A, B : list of float Vectors to be added. Returns ------- list of float Sum of `A` and `B`. """ return [a + b for a, b in zip(A, B)]
[docs] def subtract(A, B): """Calculate difference of two vectors. Parameters ---------- A, B : list of float Vectors to be subtracted. Returns ------- list of float Difference of `A` and `B`. """ return [a - b for a, b in zip(A, B)]
[docs] def dot(A, B): """Calculate dot product of two vectors. Parameters ---------- A, B : list of float Vectors to be multiplied. Returns ------- float Dot product of `A` and `B`. """ return sum(a * b for a, b in zip(A, B))
[docs] def cross(A, B): """Calculate cross product of two vectors. Parameters ---------- A, B : list of float Vectors to be multiplied. Returns ------- list of float Cross product of `A` and `B`. """ return [ A[1] * B[2] - A[2] * B[1], A[2] * B[0] - A[0] * B[2], A[0] * B[1] - A[1] * B[0], ]
[docs] def length(A): """Calculate length of vector. Parameters ---------- A : list of float Vector. Returns ------- float Length of `A`. """ return math.sqrt(dot(A, A))
[docs] def distance(A, B): """Calculate distance of two vectors. Parameters ---------- A, B : list of float Vectors. Returns ------- float Distance of `A` and `B`. """ return length(subtract(A, B))
[docs] def bonds(R1, R2=None, d1=0.0, d2=None, dmin=0.1, dmax=5.0): """Find lines that connect two sets of points. Parameters ---------- R1, R2 : list of tuple Two (ordered) sets of points. d1, d2 : float Shortening on the two line ends. dmin, dmax : float Minimum and maximum line length. Returns ------- list of list of tuple Connecting lines. """ bonds = [] if R2 is None: R2 = R1 if d2 is None: d2 = d1 oneway = R2 is R1 for n, r1 in enumerate(R1): for m, r2 in enumerate(R2): if oneway and m <= n: continue d = distance(r1, r2) if dmin < d < dmax: s1 = d1 / d s2 = d2 / d bonds.append([ [(1 - s1) * a + s1 * b for a, b in zip(r1, r2)], [s2 * a + (1 - s2) * b for a, b in zip(r1, r2)], ]) return bonds
[docs] def faces(R, d=0.0, dmin=0.1, dmax=5.0, nc=10): """Find triangular faces, e.g., of tetrahedra of atoms. Parameters ---------- R : list of tuple (Ordered) set of points. d : float Shortening at the corners, e.g., atomic radius. dmin, dmax : float Minimum and maximum side length. nc : int Number of points to trace path around corners. Returns ------- list of list of tuple Outlines of faces. """ faces = [] for i in range(len(R)): for j in range(i + 1, len(R)): if not dmin < distance(R[i], R[j]) < dmax: continue for k in range(j + 1, len(R)): if not dmin < distance(R[j], R[k]) < dmax: continue if not dmin < distance(R[k], R[i]) < dmax: continue if not d or nc < 1: face = [R[i], R[j], R[k]] else: face = [] for I, J, K in (i, j, k), (j, k, i), (k, i, j): for n in range(nc + 1): D = [(rj * n + rk * (nc - n)) / nc - ri for ri, rj, rk in zip(R[I], R[J], R[K])] face.append(add(R[I], multiply(D, d / length(D)))) face.append(face[0]) faces.append(face) return faces
[docs] def spring(r1, r2, N=500, k=50, radius=0.1, ends=0.15, xscale=1.0, yscale=1.0): """Draw coil spring in three-dimensional space. Parameters ---------- r1, r2 : list of float End points. N : int Number of path segments. k : float Winding wave vector. radius : float Radius of spring. ends : float Taper length on both ends. scalex, scaley : float, default 1.0 Scaling factors in transverse directions. If only one of these is zero, the coil becomes a flat wavy line. Returns ------- list of list of float Coordinates of spring. """ z = subtract(r2, r1) z = divide(z, length(z)) x = cross([0, 1, 0] if z[0] or z[2] else [1, 0, 0], z) x = divide(x, length(x)) y = cross(z, x) path = [] for n in range(N + 1): r = divide(add(multiply(r1, N - n), multiply(r2, n)), N) d1 = distance(r, r1) d2 = distance(r, r2) envelope = radius if d1 < ends: envelope *= (1 - math.cos(d1 * math.pi / ends)) / 2 if d2 < ends: envelope *= (1 - math.cos(d2 * math.pi / ends)) / 2 r = add(r, multiply(x, xscale * envelope * math.cos(k * d1))) r = add(r, multiply(y, yscale * envelope * math.sin(k * d1))) path.append(r) return path