Source code for tomophantom.generator

"""The set of modules to generate 2D/3D :ref:`ref_glossary_random_phantoms`. Specifically "foam"-like phantoms.
"""

import numpy as np
import math
import random


def _rand_init2D(x0min, x0max, y0min, y0max, c0min, c0max, ab_min, ab_max):
    import numpy as np

    x0 = np.random.uniform(low=x0min, high=x0max)
    y0 = np.random.uniform(low=y0min, high=y0max)
    c0 = np.random.uniform(low=c0min, high=c0max)
    ab = np.random.uniform(low=ab_min, high=ab_max)
    return (x0, y0, c0, ab)


def _rand_init3D(
    x0min, x0max, y0min, y0max, z0min, z0max, c0min, c0max, ab_min, ab_max
):
    import numpy as np

    x0 = np.random.uniform(low=x0min, high=x0max)
    y0 = np.random.uniform(low=y0min, high=y0max)
    z0 = np.random.uniform(low=z0min, high=z0max)
    c0 = np.random.uniform(low=c0min, high=c0max)
    ab = np.random.uniform(low=ab_min, high=ab_max)
    return (x0, y0, z0, c0, ab)


[docs] def foam2D( x0min, x0max, y0min, y0max, c0min, c0max, ab_min, ab_max, N_size, tot_objects, object_type, ): """ Function to generate 2D foam-like structures using different object types (ellipse, parabola, gaussian) with random parameters. """ # 2D functions from tomophantom import TomoP2D from tomophantom.TomoP2D import Objects2D attemptsNo = 2000 # the number of attempts to fit the object # objects accepted: 'ellipse', 'parabola', 'gaussian', 'mix' mix_objects = False if object_type == "ellipse": object_type = Objects2D.ELLIPSE elif object_type == "parabola": object_type = Objects2D.PARABOLA elif object_type == "gaussian": object_type = Objects2D.GAUSSIAN elif object_type == "mix": mix_objects = True else: raise TypeError("object_type can be only ellipse, parabola, gaussian or mix") X0 = np.float32(np.zeros(tot_objects)) Y0 = np.float32(np.zeros(tot_objects)) AB = np.float32(np.zeros(tot_objects)) C0_var = np.float32(np.zeros(tot_objects)) for i in range(0, tot_objects): (x0, y0, c0, ab) = _rand_init2D( x0min, x0max, y0min, y0max, c0min, c0max, ab_min, ab_max ) if i > 0: breakj = False for j in range(0, attemptsNo): if breakj == True: (x0, y0, c0, ab) = _rand_init2D( x0min, x0max, y0min, y0max, c0min, c0max, ab_min, ab_max ) breakj = False else: for l in range( 0, i ): # checks consistency with previously created objects dist = math.sqrt((X0[l] - x0) ** 2 + (Y0[l] - y0) ** 2) if (dist < (ab + AB[l])) or ( (abs(x0) + ab) ** 2 + (abs(y0) + ab) ** 2 > 1.0 ): breakj = True break if breakj == False: # re-initialise if doesn't fit the criteria X0[i] = x0 Y0[i] = y0 AB[i] = ab C0_var[i] = c0 break if AB[i] == 0.0: X0[i] = x0 Y0[i] = y0 AB[i] = 0.0001 C0_var[i] = c0 myObjects = [] # dictionary of objects for obj in range(0, len(X0)): if mix_objects == True: rand_obj = random.randint(0, 2) if rand_obj == 0: object_type = Objects2D.ELLIPSE if rand_obj == 1: object_type = Objects2D.PARABOLA if rand_obj == 2: object_type = Objects2D.GAUSSIAN curr_obj = { "Obj": object_type, "C0": C0_var[obj], "x0": X0[obj], "y0": Y0[obj], "a": AB[obj], "b": AB[obj], "phi": 0.0, } myObjects.append(curr_obj) Object = TomoP2D.Object(N_size, myObjects) return (Object, myObjects)
# Function to generate 3D foam-like structures using randomly located spheres
[docs] def foam3D( x0min, x0max, y0min, y0max, z0min, z0max, c0min, c0max, ab_min, ab_max, N_size, tot_objects, object_type, ): """ Function to generate 3D foam-like structures using different object types (ellipsoids, paraboloids, gaussians) with random parameters. """ # 3D functions from tomophantom import TomoP3D from tomophantom.TomoP3D import Objects3D attemptsNo = 2000 # objects accepted: 'ellipsoid', 'paraboloid', 'gaussian', 'mix' mix_objects = False if object_type == "ellipsoid": object_type = Objects3D.ELLIPSOID elif object_type == "paraboloid": object_type = Objects3D.PARABOLOID elif object_type == "gaussian": object_type = Objects3D.GAUSSIAN elif object_type == "mix": mix_objects = True else: raise TypeError("object_type can be only ellipse, parabola, gaussian or mix") X0 = np.float32(np.zeros(tot_objects)) Y0 = np.float32(np.zeros(tot_objects)) Z0 = np.float32(np.zeros(tot_objects)) AB = np.float32(np.zeros(tot_objects)) C0_var = np.float32(np.zeros(tot_objects)) for i in range(0, tot_objects): (x0, y0, z0, c0, ab) = _rand_init3D( x0min, x0max, y0min, y0max, z0min, z0max, c0min, c0max, ab_min, ab_max ) if i > 0: breakj = False for j in range(0, attemptsNo): if breakj: (x0, y0, z0, c0, ab) = _rand_init3D( x0min, x0max, y0min, y0max, z0min, z0max, c0min, c0max, ab_min, ab_max, ) breakj = False else: for l in range( 0, i ): # checks consistency with previously created objects dist = math.sqrt( (X0[l] - x0) ** 2 + (Y0[l] - y0) ** 2 + (Z0[l] - z0) ** 2 ) if (dist < (ab + AB[l])) or ( (abs(x0) + ab) ** 2 + (abs(y0) + ab) ** 2 + (abs(z0) + ab) ** 2 > 1.0 ): breakj = True break if breakj == False: # re-initialise if doesn't fit the criteria X0[i] = x0 Y0[i] = y0 Z0[i] = z0 AB[i] = ab C0_var[i] = c0 break if AB[i] == 0.0: X0[i] = x0 Y0[i] = y0 AB[i] = 0.0001 C0_var[i] = c0 myObjects = [] # dictionary of objects for obj in range(0, len(X0)): if mix_objects == True: rand_obj = random.randint(0, 2) if rand_obj == 0: object_type = Objects3D.ELLIPSOID if rand_obj == 1: object_type = Objects3D.PARABOLOID if rand_obj == 2: object_type = Objects3D.GAUSSIAN curr_obj = { "Obj": object_type, "C0": C0_var[obj], "x0": X0[obj], "y0": Y0[obj], "z0": Z0[obj], "a": AB[obj], "b": AB[obj], "c": AB[obj], "phi1": 0.0, } myObjects.append(curr_obj) Object3D = TomoP3D.Object(N_size, myObjects) return (Object3D, myObjects)