DAMASK_EICMD/python/damask/_geomgrid.py

1480 lines
57 KiB
Python
Raw Normal View History

2021-06-15 23:08:01 +05:30
import os
2020-08-22 23:25:18 +05:30
import copy
2021-06-15 23:08:01 +05:30
import warnings
2020-09-18 20:02:08 +05:30
import multiprocessing as mp
2020-03-29 22:42:23 +05:30
from functools import partial
2022-01-13 01:04:29 +05:30
import typing
from typing import Optional, Union, TextIO, Sequence, Dict
2021-12-06 18:52:52 +05:30
from pathlib import Path
2019-05-26 15:33:21 +05:30
import numpy as np
import pandas as pd
import h5py
2022-11-09 00:22:08 +05:30
from scipy import ndimage, spatial, interpolate
2019-05-26 15:33:21 +05:30
2020-03-11 12:02:03 +05:30
from . import VTK
2019-05-28 01:30:26 +05:30
from . import util
2020-03-29 22:42:23 +05:30
from . import grid_filters
from . import Rotation
2021-12-06 18:52:52 +05:30
from . import Table
2022-02-21 15:49:53 +05:30
from . import Colormap
from ._typehints import FloatSequence, IntSequence, NumpyRngSeed
try:
import numba as nb # type: ignore
except ImportError:
nb = False
def numba_njit_wrapper(**kwargs):
return (lambda function: nb.njit(function) if nb else function)
2019-05-26 15:33:21 +05:30
2023-11-28 03:11:28 +05:30
class GeomGrid:
"""
Geometry definition for grid solvers.
2023-02-21 20:57:06 +05:30
Create and manipulate geometry definitions for storage as VTK ImageData
files ('.vti' extension). A grid has a physical size, a coordinate origin,
and contains the material ID (indexing an entry in 'material.yaml')
as well as initial condition fields.
"""
2019-11-23 01:22:36 +05:30
2021-12-06 18:52:52 +05:30
def __init__(self,
material: np.ndarray,
2022-01-13 01:04:29 +05:30
size: FloatSequence,
origin: FloatSequence = np.zeros(3),
2022-11-23 02:56:15 +05:30
initial_conditions: Optional[Dict[str,np.ndarray]] = None,
comments: Union[None, str, Sequence[str]] = None):
2019-11-23 01:22:36 +05:30
"""
New geometry definition for grid solvers.
2019-11-23 01:22:36 +05:30
Parameters
----------
2022-11-09 00:22:08 +05:30
material : numpy.ndarray of int, shape (:,:,:)
Material indices. The shape of the material array defines
the number of cells.
2022-01-13 01:04:29 +05:30
size : sequence of float, len (3)
2021-06-17 21:56:37 +05:30
Physical size of grid in meter.
2022-01-13 01:04:29 +05:30
origin : sequence of float, len (3), optional
Coordinates of grid origin in meter. Defaults to [0.0,0.0,0.0].
2022-03-20 01:19:33 +05:30
initial_conditions : dictionary, optional
2023-02-21 20:57:06 +05:30
Initial condition label and field values at each grid point.
comments : (sequence of) str, optional
Additional, human-readable information, e.g. history of operations.
2019-11-23 01:22:36 +05:30
"""
self.material = material
self.size = size # type: ignore
self.origin = origin # type: ignore
self.initial_conditions = {} if initial_conditions is None else initial_conditions
self.comments = [] if comments is None else \
[comments] if isinstance(comments,str) else \
[str(c) for c in comments]
2021-12-06 18:52:52 +05:30
def __repr__(self) -> str:
"""
Return repr(self).
2023-02-21 20:57:06 +05:30
Give short, human-readable summary.
"""
mat_min = np.nanmin(self.material)
mat_max = np.nanmax(self.material)
mat_N = self.N_materials
2019-11-23 01:22:36 +05:30
return util.srepr([
2022-01-29 20:29:14 +05:30
f'cells: {util.srepr(self.cells, " × ")}',
2022-02-17 11:43:39 +05:30
f'size: {util.srepr(self.size, " × ")}',
f'origin: {util.srepr(self.origin," ")} m',
2020-12-11 01:23:11 +05:30
f'# materials: {mat_N}' + ('' if mat_min == 0 and mat_max+1 == mat_N else
f' (min: {mat_min}, max: {mat_max})')
]+(['initial_conditions:']+[f' - {f}' for f in self.initial_conditions] if self.initial_conditions else []))
2019-11-23 01:22:36 +05:30
2023-11-28 03:11:28 +05:30
def __copy__(self) -> 'GeomGrid':
"""
Return deepcopy(self).
Create deep copy.
"""
2020-08-22 23:25:18 +05:30
return copy.deepcopy(self)
2021-01-03 16:33:40 +05:30
copy = __copy__
2020-08-22 23:25:18 +05:30
2022-01-27 04:07:07 +05:30
def __eq__(self,
other: object) -> bool:
"""
Return self==other.
Test equality of other.
Parameters
----------
2023-11-28 03:11:28 +05:30
other : damask.GeomGrid
GeomGrid to compare self against.
"""
2023-11-28 03:11:28 +05:30
if not isinstance(other, GeomGrid):
2022-01-13 01:04:29 +05:30
return NotImplemented
return bool( np.allclose(other.size,self.size)
and np.allclose(other.origin,self.origin)
and np.all(other.cells == self.cells)
and np.all(other.material == self.material))
2019-11-23 01:22:36 +05:30
@property
2021-12-06 18:52:52 +05:30
def material(self) -> np.ndarray:
"""Material indices."""
return self._material
@material.setter
def material(self,
material: np.ndarray):
if len(material.shape) != 3:
raise ValueError(f'invalid material shape {material.shape}')
if material.dtype not in np.sctypes['float'] and material.dtype not in np.sctypes['int']:
raise TypeError(f'invalid material data type "{material.dtype}"')
self._material = np.copy(material)
if self.material.dtype in np.sctypes['float'] and \
np.all(self.material == self.material.astype(np.int64).astype(float)):
self._material = self.material.astype(np.int64)
@property
2021-12-06 18:52:52 +05:30
def size(self) -> np.ndarray:
2023-02-21 20:57:06 +05:30
"""Edge lengths of grid in meter."""
return self._size
@size.setter
def size(self,
size: FloatSequence):
if len(size) != 3 or any(np.array(size) < 0):
raise ValueError(f'invalid size {size}')
self._size = np.array(size)
@property
2022-01-13 01:04:29 +05:30
def origin(self) -> np.ndarray:
2023-02-21 20:57:06 +05:30
"""Vector to grid origin in meter."""
return self._origin
@origin.setter
def origin(self,
origin: FloatSequence):
if len(origin) != 3:
raise ValueError(f'invalid origin {origin}')
self._origin = np.array(origin)
@property
def initial_conditions(self) -> Dict[str,np.ndarray]:
"""Fields of initial conditions."""
2022-04-22 23:09:23 +05:30
self._ic = dict(zip(self._ic.keys(), # type: ignore
[v if isinstance(v,np.ndarray) else
np.broadcast_to(v,self.cells) for v in self._ic.values()])) # type: ignore
return self._ic
@initial_conditions.setter
def initial_conditions(self,
ic: Dict[str,np.ndarray]):
if not isinstance(ic,dict):
raise TypeError('initial conditions is not a dictionary')
self._ic = ic
2019-12-08 13:47:57 +05:30
@property
2021-12-06 18:52:52 +05:30
def cells(self) -> np.ndarray:
2023-02-21 20:57:06 +05:30
"""Cell counts along x,y,z direction."""
return np.asarray(self.material.shape)
2019-11-23 01:22:36 +05:30
2020-03-18 18:19:53 +05:30
@property
2021-12-06 18:52:52 +05:30
def N_materials(self) -> int:
2020-12-04 11:42:18 +05:30
"""Number of (unique) material indices within grid."""
return np.unique(self.material).size
2019-11-23 01:22:36 +05:30
@staticmethod
def _load(fname: Union[str, Path],label) -> 'GeomGrid':
"""
2023-02-21 20:57:06 +05:30
Load from VTK ImageData file.
Parameters
----------
2022-01-13 01:04:29 +05:30
fname : str or pathlib.Path
VTK ImageData file to read.
Valid extension is .vti, which will be appended if not given.
label : str
Label of the dataset containing the material IDs.
Returns
-------
2023-11-28 03:11:28 +05:30
loaded : damask.GeomGrid
GeomGrid-based geometry from file.
"""
2022-02-11 03:43:37 +05:30
v = VTK.load(fname if str(fname).endswith('.vti') else str(fname)+'.vti')
2020-12-04 02:28:24 +05:30
cells = np.array(v.vtk_data.GetDimensions())-1
bbox = np.array(v.vtk_data.GetBounds()).reshape(3,2).T
ic = {label:v.get(l).reshape(cells,order='F') for l in set(v.labels['Cell Data']) - {label}}
return GeomGrid(material = v.get(label).reshape(cells,order='F'),
2023-11-28 03:11:28 +05:30
size = bbox[1] - bbox[0],
origin = bbox[0],
initial_conditions = ic,
comments = v.comments,
)
@staticmethod
def load(fname: Union[str, Path]) -> 'GeomGrid':
"""
Load from VTK ImageData file with material IDs stored as 'material'.
Parameters
----------
fname : str or pathlib.Path
GeomGrid file to read.
Valid extension is .vti, which will be appended if not given.
Returns
-------
loaded : damask.GeomGrid
GeomGrid-based geometry from file.
"""
return GeomGrid._load(fname,'material')
@staticmethod
def load_SPPARKS(fname: Union[str, Path]) -> 'GeomGrid':
"""
Load from SPPARKs VTK dump.
Parameters
----------
fname : str or pathlib.Path
SPPARKS VTK dump file to read.
Valid extension is .vti, which will be appended if not given.
Returns
-------
loaded : damask.GeomGrid
GeomGrid-based geometry from file.
Notes
-----
A SPPARKs VTI dump is equivalent to a DAMASK VTI file
where 'material' is renamed to 'spins'.
"""
return GeomGrid._load(fname,'spins')
@typing.no_type_check
@staticmethod
2023-11-28 03:11:28 +05:30
def load_ASCII(fname)-> 'GeomGrid':
2019-11-23 01:22:36 +05:30
"""
2020-12-04 02:28:24 +05:30
Load from geom file.
2019-11-23 01:22:36 +05:30
2020-11-14 22:24:47 +05:30
Storing geometry files in ASCII format is deprecated.
This function will be removed in a future version of DAMASK.
2019-11-23 01:22:36 +05:30
Parameters
----------
fname : str, pathlib.Path, or file handle
2020-08-08 23:12:34 +05:30
Geometry file to read.
2019-11-23 01:22:36 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
loaded : damask.GeomGrid
GeomGrid-based geometry from file.
2019-11-23 01:22:36 +05:30
"""
2022-01-05 18:36:06 +05:30
warnings.warn('Support for ASCII-based geom format will be removed in DAMASK 3.0.0', DeprecationWarning,2)
2021-12-06 18:52:52 +05:30
if isinstance(fname, (str, Path)):
f = open(fname)
2021-12-06 18:52:52 +05:30
elif isinstance(fname, TextIO):
f = fname
2021-12-06 18:52:52 +05:30
else:
raise TypeError
f.seek(0)
try:
2021-12-06 18:52:52 +05:30
header_length_,keyword = f.readline().split()[:2]
header_length = int(header_length_)
except ValueError:
header_length,keyword = (-1, 'invalid')
2019-11-23 01:22:36 +05:30
if not keyword.startswith('head') or header_length < 3:
raise TypeError('invalid or missing header length information')
2019-11-23 01:22:36 +05:30
2020-03-21 15:37:21 +05:30
comments = []
2020-11-20 00:56:15 +05:30
content = f.readlines()
2019-11-23 01:22:36 +05:30
for i,line in enumerate(content[:header_length]):
2022-01-13 01:04:29 +05:30
items = line.split('#')[0].lower().strip().split()
if (key := items[0] if items else '') == 'grid':
2022-01-13 01:04:29 +05:30
cells = np.array([ int(dict(zip(items[1::2],items[2::2]))[i]) for i in ['a','b','c']])
2019-11-23 01:22:36 +05:30
elif key == 'size':
size = np.array([float(dict(zip(items[1::2],items[2::2]))[i]) for i in ['x','y','z']])
elif key == 'origin':
origin = np.array([float(dict(zip(items[1::2],items[2::2]))[i]) for i in ['x','y','z']])
else:
comments.append(line.strip())
material = np.empty(cells.prod()) # initialize as flat array
2019-11-23 01:22:36 +05:30
i = 0
for line in content[header_length:]:
if len(items := line.split('#')[0].split()) == 3:
2021-12-06 18:52:52 +05:30
if items[1].lower() == 'of':
material_entry = np.ones(int(items[0]))*float(items[2])
2019-11-23 01:22:36 +05:30
elif items[1].lower() == 'to':
2021-12-06 18:52:52 +05:30
material_entry = np.linspace(int(items[0]),int(items[2]),
2019-11-23 01:22:36 +05:30
abs(int(items[2])-int(items[0]))+1,dtype=float)
2021-12-06 18:52:52 +05:30
else: material_entry = list(map(float, items))
else: material_entry = list(map(float, items))
material[i:i+len(material_entry)] = material_entry
2019-11-23 01:22:36 +05:30
i += len(items)
2020-03-21 15:37:21 +05:30
2020-12-04 02:28:24 +05:30
if i != cells.prod():
raise TypeError(f'mismatch between {cells.prod()} expected entries and {i} found')
2020-03-21 15:37:21 +05:30
if not np.any(np.mod(material,1) != 0.0): # no float present
material = material.astype(np.int64) - (1 if material.min() > 0 else 0)
2020-03-21 15:37:21 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = material.reshape(cells,order='F'),
size = size,
origin = origin,
comments = comments,
)
2019-11-23 01:22:36 +05:30
2021-06-16 00:53:10 +05:30
@staticmethod
2023-11-28 03:11:28 +05:30
def load_Neper(fname: Union[str, Path]) -> 'GeomGrid':
2021-06-16 00:53:10 +05:30
"""
Load from Neper VTK file.
Parameters
----------
2022-01-13 01:04:29 +05:30
fname : str or pathlib.Path
2021-06-16 00:53:10 +05:30
Geometry file to read.
Returns
-------
2023-11-28 03:11:28 +05:30
loaded : damask.GeomGrid
GeomGrid-based geometry from file.
2021-06-16 00:53:10 +05:30
Notes
-----
Material indices in Neper usually start at 1 unless
a buffer material with index 0 is added.
2022-03-04 21:47:03 +05:30
Examples
--------
Read a periodic polycrystal generated with Neper.
>>> import damask
>>> N_grains = 20
>>> cells = (32,32,32)
2022-04-22 23:57:03 +05:30
>>> damask.util.run(f'neper -T -n {N_grains} -tesrsize {cells[0]}:{cells[1]}:{cells[2]} -periodicity all -format vtk')
2023-11-28 03:11:28 +05:30
>>> damask.GeomGrid.load_Neper(f'n{N_grains}-id1.vtk').renumber()
2022-03-04 21:47:03 +05:30
cells: 32 × 32 × 32
size: 1.0 × 1.0 × 1.0
origin: 0.0 0.0 0.0 m
# materials: 20
2021-06-16 00:53:10 +05:30
"""
2022-01-22 12:35:49 +05:30
v = VTK.load(fname,'ImageData')
2021-06-16 00:53:10 +05:30
cells = np.array(v.vtk_data.GetDimensions())-1
bbox = np.array(v.vtk_data.GetBounds()).reshape(3,2).T
2023-11-28 03:11:28 +05:30
return GeomGrid(material = v.get('MaterialId').reshape(cells,order='F').astype('int32',casting='unsafe'),
size = bbox[1] - bbox[0],
origin = bbox[0],
comments = util.execution_stamp('GeomGrid','load_Neper'),
)
2021-06-16 00:53:10 +05:30
2020-10-09 17:49:19 +05:30
@staticmethod
2022-01-13 01:04:29 +05:30
def load_DREAM3D(fname: Union[str, Path],
2022-11-23 02:56:15 +05:30
feature_IDs: Optional[str] = None,
cell_data: Optional[str] = None,
phases: str = 'Phases',
Euler_angles: str = 'EulerAngles',
2023-11-28 03:11:28 +05:30
base_group: Optional[str] = None) -> 'GeomGrid':
2020-10-09 17:49:19 +05:30
"""
2021-03-20 18:07:06 +05:30
Load DREAM.3D (HDF5) file.
2023-09-22 17:41:25 +05:30
Data in DREAM.3D files can be stored per cell ('CellData')
and/or per grain ('Grain Data'). Per default, i.e. if
'feature_IDs' is None, cell-wise data is assumed.
2021-03-20 18:07:06 +05:30
2020-10-09 17:49:19 +05:30
Parameters
----------
2023-09-22 18:45:52 +05:30
fname : str or pathlib.Path
Filename of the DREAM.3D (HDF5) file.
2022-01-13 01:04:29 +05:30
feature_IDs : str, optional
2021-03-20 18:07:06 +05:30
Name of the dataset containing the mapping between cells and
grain-wise data. Defaults to 'None', in which case cell-wise
data is used.
2022-01-13 01:04:29 +05:30
cell_data : str, optional
2021-03-23 18:58:56 +05:30
Name of the group (folder) containing cell-wise data. Defaults to
None in wich case it is automatically detected.
2022-01-13 01:04:29 +05:30
phases : str, optional
2021-03-20 18:07:06 +05:30
Name of the dataset containing the phase ID. It is not used for
grain-wise data, i.e. when feature_IDs is not None.
Defaults to 'Phases'.
2022-01-13 01:04:29 +05:30
Euler_angles : str, optional
2021-03-20 18:07:06 +05:30
Name of the dataset containing the crystallographic orientation as
Euler angles in radians It is not used for grain-wise data, i.e.
when feature_IDs is not None. Defaults to 'EulerAngles'.
2022-01-13 01:04:29 +05:30
base_group : str, optional
2021-03-20 18:07:06 +05:30
Path to the group (folder) that contains geometry (_SIMPL_GEOMETRY),
and grain- or cell-wise data. Defaults to None, in which case
2021-03-20 04:19:41 +05:30
it is set as the path that contains _SIMPL_GEOMETRY/SPACING.
Returns
-------
2023-11-28 03:11:28 +05:30
loaded : damask.GeomGrid
GeomGrid-based geometry from file.
2021-03-20 18:07:06 +05:30
2023-09-22 17:41:25 +05:30
Notes
-----
damask.ConfigMaterial.load_DREAM3D gives the corresponding
material definition.
For cell-wise data, only unique combinations of
orientation and phase are considered.
2020-10-09 17:49:19 +05:30
"""
2023-09-22 18:45:52 +05:30
with h5py.File(fname, 'r') as f:
b = util.DREAM3D_base_group(f) if base_group is None else base_group
c = util.DREAM3D_cell_data_group(f) if cell_data is None else cell_data
cells = f['/'.join([b,'_SIMPL_GEOMETRY','DIMENSIONS'])][()]
size = f['/'.join([b,'_SIMPL_GEOMETRY','SPACING'])] * cells
origin = f['/'.join([b,'_SIMPL_GEOMETRY','ORIGIN'])][()]
if feature_IDs is None:
phase = f['/'.join([b,c,phases])][()].reshape(-1,1)
O = Rotation.from_Euler_angles(f['/'.join([b,c,Euler_angles])]).as_quaternion().reshape(-1,4) # noqa
unique,unique_inverse = np.unique(np.hstack([O,phase]),return_inverse=True,axis=0)
ma = np.arange(cells.prod()) if len(unique) == cells.prod() else \
np.arange(unique.size)[np.argsort(pd.unique(unique_inverse))][unique_inverse]
else:
ma = f['/'.join([b,c,feature_IDs])][()].flatten()
2020-10-09 17:49:19 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = ma.reshape(cells,order='F'),
size = size,
origin = origin,
comments = util.execution_stamp('GeomGrid','load_DREAM3D'),
)
2020-10-09 17:49:19 +05:30
@staticmethod
2022-01-13 01:04:29 +05:30
def from_table(table: Table,
coordinates: str,
2023-11-28 03:11:28 +05:30
labels: Union[str, Sequence[str]]) -> 'GeomGrid':
2020-10-09 17:49:19 +05:30
"""
Create grid from ASCII table.
2020-10-09 17:49:19 +05:30
Parameters
----------
table : damask.Table
Table that contains material information.
2020-10-09 17:49:19 +05:30
coordinates : str
Label of the vector column containing the spatial coordinates.
Need to be ordered (1./x fast, 3./z slow).
labels : (sequence of) str
2020-10-09 17:49:19 +05:30
Label(s) of the columns containing the material definition.
Each unique combination of values results in one material ID.
2020-10-09 17:49:19 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
new : damask.GeomGrid
GeomGrid-based geometry from values in table.
2020-10-09 17:49:19 +05:30
"""
cells,size,origin = grid_filters.cellsSizeOrigin_coordinates0_point(table.get(coordinates))
2020-10-09 17:49:19 +05:30
labels_ = [labels] if isinstance(labels,str) else labels
unique,unique_inverse = np.unique(np.hstack([table.get(l) for l in labels_]),return_inverse=True,axis=0)
2020-10-31 21:53:58 +05:30
2020-12-04 02:28:24 +05:30
ma = np.arange(cells.prod()) if len(unique) == cells.prod() else \
2020-10-31 21:53:58 +05:30
np.arange(unique.size)[np.argsort(pd.unique(unique_inverse))][unique_inverse]
2020-10-09 17:49:19 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = ma.reshape(cells,order='F'),
size = size,
origin = origin,
comments = util.execution_stamp('GeomGrid','from_table'),
)
2020-10-09 17:49:19 +05:30
2020-03-29 22:42:23 +05:30
@staticmethod
def _find_closest_seed(seeds: np.ndarray,
weights: np.ndarray,
point: np.ndarray) -> np.integer:
2020-03-29 22:42:23 +05:30
return np.argmin(np.sum((np.broadcast_to(point,(len(seeds),3))-seeds)**2,axis=1) - weights)
2020-06-24 21:35:12 +05:30
2020-03-29 22:42:23 +05:30
@staticmethod
2022-01-13 01:04:29 +05:30
def from_Laguerre_tessellation(cells: IntSequence,
size: FloatSequence,
seeds: np.ndarray,
weights: FloatSequence,
2022-11-23 02:56:15 +05:30
material: Optional[IntSequence] = None,
2022-01-13 01:04:29 +05:30
periodic: bool = True):
2020-03-29 22:42:23 +05:30
"""
Create grid from Laguerre tessellation.
2020-03-29 22:42:23 +05:30
Parameters
----------
2022-01-13 01:04:29 +05:30
cells : sequence of int, len (3)
2023-02-21 20:57:06 +05:30
Cell counts along x,y,z direction.
2022-01-13 01:04:29 +05:30
size : sequence of float, len (3)
2023-02-21 20:57:06 +05:30
Edge lengths of the grid in meter.
2022-11-09 00:22:08 +05:30
seeds : numpy.ndarray of float, shape (:,3)
2023-10-13 02:51:40 +05:30
Position of the seed points in meter. All points need
to lay within the box [(0,0,0),size].
2022-01-13 01:04:29 +05:30
weights : sequence of float, len (seeds.shape[0])
2023-10-13 02:51:40 +05:30
Weights of the seeds. Setting all weights to 1.0 gives a
standard Voronoi tessellation.
2022-01-13 01:04:29 +05:30
material : sequence of int, len (seeds.shape[0]), optional
Material ID of the seeds.
Defaults to None, in which case materials are consecutively numbered.
2022-01-13 03:43:38 +05:30
periodic : bool, optional
2020-12-04 11:42:18 +05:30
Assume grid to be periodic. Defaults to True.
2020-03-29 22:42:23 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
new : damask.GeomGrid
GeomGrid-based geometry from tessellation.
2020-03-29 22:42:23 +05:30
"""
weights_p: FloatSequence
2020-03-29 22:42:23 +05:30
if periodic:
2020-04-20 23:46:25 +05:30
weights_p = np.tile(weights,27) # Laguerre weights (1,2,3,1,2,3,...,1,2,3)
2020-03-29 22:42:23 +05:30
seeds_p = np.vstack((seeds -np.array([size[0],0.,0.]),seeds, seeds +np.array([size[0],0.,0.])))
seeds_p = np.vstack((seeds_p-np.array([0.,size[1],0.]),seeds_p,seeds_p+np.array([0.,size[1],0.])))
seeds_p = np.vstack((seeds_p-np.array([0.,0.,size[2]]),seeds_p,seeds_p+np.array([0.,0.,size[2]])))
else:
2022-01-20 17:22:56 +05:30
weights_p = np.array(weights,float)
2020-03-29 22:42:23 +05:30
seeds_p = seeds
coords = grid_filters.coordinates0_point(cells,size).reshape(-1,3)
2022-01-13 01:04:29 +05:30
pool = mp.Pool(int(os.environ.get('OMP_NUM_THREADS',4)))
2023-11-28 03:11:28 +05:30
result = pool.map_async(partial(GeomGrid._find_closest_seed,seeds_p,weights_p), coords)
2020-03-29 22:42:23 +05:30
pool.close()
pool.join()
material_ = np.array(result.get()).reshape(cells)
2020-03-29 22:42:23 +05:30
if periodic: material_ %= len(weights)
2020-03-29 22:42:23 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = material_ if material is None else np.array(material)[material_],
size = size,
comments = util.execution_stamp('GeomGrid','from_Laguerre_tessellation'),
)
2020-03-29 22:42:23 +05:30
@staticmethod
2022-01-13 01:04:29 +05:30
def from_Voronoi_tessellation(cells: IntSequence,
size: FloatSequence,
2021-12-06 18:52:52 +05:30
seeds: np.ndarray,
2022-11-23 02:56:15 +05:30
material: Optional[IntSequence] = None,
2023-11-28 03:11:28 +05:30
periodic: bool = True) -> 'GeomGrid':
2020-03-29 22:42:23 +05:30
"""
Create grid from Voronoi tessellation.
2020-03-29 22:42:23 +05:30
Parameters
----------
2022-01-13 01:04:29 +05:30
cells : sequence of int, len (3)
2023-02-21 20:57:06 +05:30
Cell counts along x,y,z direction.
2022-01-13 01:04:29 +05:30
size : sequence of float, len (3)
2023-02-21 20:57:06 +05:30
Edge lengths of the grid in meter.
2022-11-09 00:22:08 +05:30
seeds : numpy.ndarray of float, shape (:,3)
2023-10-13 02:51:40 +05:30
Position of the seed points in meter. All points need
to lay within the box [(0,0,0),size].
2022-01-13 01:04:29 +05:30
material : sequence of int, len (seeds.shape[0]), optional
Material ID of the seeds.
Defaults to None, in which case materials are consecutively numbered.
2022-01-13 03:43:38 +05:30
periodic : bool, optional
2020-12-04 11:42:18 +05:30
Assume grid to be periodic. Defaults to True.
2020-03-29 22:42:23 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
new : damask.GeomGrid
GeomGrid-based geometry from tessellation.
2020-03-29 22:42:23 +05:30
"""
coords = grid_filters.coordinates0_point(cells,size).reshape(-1,3)
2021-04-29 12:26:40 +05:30
tree = spatial.cKDTree(seeds,boxsize=size) if periodic else \
spatial.cKDTree(seeds)
try:
material_ = tree.query(coords, workers = int(os.environ.get('OMP_NUM_THREADS',4)))[1]
except TypeError:
material_ = tree.query(coords, n_jobs = int(os.environ.get('OMP_NUM_THREADS',4)))[1] # scipy <1.6
2020-03-29 22:42:23 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = (material_ if material is None else np.array(material)[material_]).reshape(cells),
size = size,
comments = util.execution_stamp('GeomGrid','from_Voronoi_tessellation'),
)
2020-03-29 22:42:23 +05:30
_minimal_surface = \
{'Schwarz P': lambda x,y,z: np.cos(x) + np.cos(y) + np.cos(z),
'Double Primitive': lambda x,y,z: ( 0.5 * (np.cos(x)*np.cos(y) + np.cos(y)*np.cos(z) + np.cos(z)*np.cos(x))
+ 0.2 * (np.cos(2*x) + np.cos(2*y) + np.cos(2*z)) ),
'Schwarz D': lambda x,y,z: ( np.sin(x)*np.sin(y)*np.sin(z)
+ np.sin(x)*np.cos(y)*np.cos(z)
+ np.cos(x)*np.cos(y)*np.sin(z)
+ np.cos(x)*np.sin(y)*np.cos(z) ),
'Complementary D': lambda x,y,z: ( np.cos(3*x+y)*np.cos(z) - np.sin(3*x-y)*np.sin(z) + np.cos(x+3*y)*np.cos(z)
+ np.sin(x-3*y)*np.sin(z) + np.cos(x-y)*np.cos(3*z) - np.sin(x+y)*np.sin(3*z) ),
'Double Diamond': lambda x,y,z: 0.5 * (np.sin(x)*np.sin(y)
+ np.sin(y)*np.sin(z)
+ np.sin(z)*np.sin(x)
+ np.cos(x) * np.cos(y) * np.cos(z) ),
'Dprime': lambda x,y,z: 0.5 * ( np.cos(x)*np.cos(y)*np.cos(z)
+ np.cos(x)*np.sin(y)*np.sin(z)
+ np.sin(x)*np.cos(y)*np.sin(z)
+ np.sin(x)*np.sin(y)*np.cos(z)
- np.sin(2*x)*np.sin(2*y)
- np.sin(2*y)*np.sin(2*z)
- np.sin(2*z)*np.sin(2*x) ) - 0.2,
'Gyroid': lambda x,y,z: np.cos(x)*np.sin(y) + np.cos(y)*np.sin(z) + np.cos(z)*np.sin(x),
'Gprime': lambda x,y,z : ( np.sin(2*x)*np.cos(y)*np.sin(z)
+ np.sin(2*y)*np.cos(z)*np.sin(x)
+ np.sin(2*z)*np.cos(x)*np.sin(y) ) + 0.32,
'Karcher K': lambda x,y,z: ( 0.3 * ( np.cos(x) + np.cos(y) + np.cos(z)
+ np.cos(x)*np.cos(y) + np.cos(y)*np.cos(z) + np.cos(z)*np.cos(x) )
- 0.4 * ( np.cos(2*x) + np.cos(2*y) + np.cos(2*z) ) ) + 0.2,
'Lidinoid': lambda x,y,z: 0.5 * ( np.sin(2*x)*np.cos(y)*np.sin(z)
+ np.sin(2*y)*np.cos(z)*np.sin(x)
+ np.sin(2*z)*np.cos(x)*np.sin(y)
- np.cos(2*x)*np.cos(2*y)
- np.cos(2*y)*np.cos(2*z)
- np.cos(2*z)*np.cos(2*x) ) + 0.15,
'Neovius': lambda x,y,z: ( 3 * (np.cos(x)+np.cos(y)+np.cos(z))
+ 4 * np.cos(x)*np.cos(y)*np.cos(z) ),
'Fisher-Koch S': lambda x,y,z: ( np.cos(2*x)*np.sin( y)*np.cos( z)
+ np.cos( x)*np.cos(2*y)*np.sin( z)
+ np.sin( x)*np.cos( y)*np.cos(2*z) ),
}
2020-09-30 10:40:15 +05:30
@staticmethod
2022-01-13 01:04:29 +05:30
def from_minimal_surface(cells: IntSequence,
size: FloatSequence,
2021-12-06 18:52:52 +05:30
surface: str,
threshold: float = 0.0,
periods: int = 1,
2023-11-28 03:11:28 +05:30
materials: IntSequence = (0,1)) -> 'GeomGrid':
2020-09-30 10:40:15 +05:30
"""
2023-02-21 20:57:06 +05:30
Create grid from definition of triply-periodic minimal surface.
2020-09-30 10:40:15 +05:30
Parameters
----------
2022-01-13 01:04:29 +05:30
cells : sequence of int, len (3)
2023-02-21 20:57:06 +05:30
Cell counts along x,y,z direction.
2022-01-13 01:04:29 +05:30
size : sequence of float, len (3)
2023-02-21 20:57:06 +05:30
Edge lengths of the grid in meter.
surface : str
Type of the minimal surface. See notes for details.
2020-09-30 10:40:15 +05:30
threshold : float, optional.
Threshold of the minimal surface. Defaults to 0.0.
periods : integer, optional.
Number of periods per unit cell. Defaults to 1.
2022-01-13 01:04:29 +05:30
materials : sequence of int, len (2)
Material IDs. Defaults to (0,1).
2020-10-28 14:01:55 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
new : damask.GeomGrid
GeomGrid-based geometry from definition of minimal surface.
Notes
-----
The following triply-periodic minimal surfaces are implemented:
- Schwarz P
- Double Primitive
- Schwarz D
- Complementary D
- Double Diamond
- Dprime
- Gyroid
- Gprime
- Karcher K
- Lidinoid
- Neovius
- Fisher-Koch S
2020-09-30 10:40:15 +05:30
References
----------
2021-03-18 20:06:40 +05:30
S.B.G. Blanquer et al., Biofabrication 9(2):025001, 2017
https://doi.org/10.1088/1758-5090/aa6553
2021-03-18 20:06:40 +05:30
M. Wohlgemuth et al., Macromolecules 34(17):6083-6089, 2001
https://doi.org/10.1021/ma0019499
2021-03-18 20:06:40 +05:30
M.-T. Hsieh and L. Valdevit, Software Impacts 6:100026, 2020
https://doi.org/10.1016/j.simpa.2020.100026
Examples
--------
Minimal surface of 'Gyroid' type.
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> damask.GeomGrid.from_minimal_surface([64]*3,np.ones(3)*1.e-4,'Gyroid')
2023-02-21 20:57:06 +05:30
cells : 64 × 64 × 64
size : 0.0001 × 0.0001 × 0.0001
2022-02-17 11:43:39 +05:30
origin: 0.0 0.0 0.0 m
# materials: 2
2023-02-21 20:57:06 +05:30
Minimal surface of 'Neovius' type with specific material IDs.
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> damask.GeomGrid.from_minimal_surface([80]*3,np.ones(3)*5.e-4,
... 'Neovius',materials=(1,5))
2023-02-21 20:57:06 +05:30
cells : 80 × 80 × 80
size : 0.0005 × 0.0005 × 0.0005
2022-02-17 11:43:39 +05:30
origin: 0.0 0.0 0.0 m
# materials: 2 (min: 1, max: 5)
2020-09-30 10:40:15 +05:30
"""
2020-12-04 02:28:24 +05:30
x,y,z = np.meshgrid(periods*2.0*np.pi*(np.arange(cells[0])+0.5)/cells[0],
periods*2.0*np.pi*(np.arange(cells[1])+0.5)/cells[1],
periods*2.0*np.pi*(np.arange(cells[2])+0.5)/cells[2],
2020-09-30 10:40:15 +05:30
indexing='ij',sparse=True)
2023-11-28 03:11:28 +05:30
return GeomGrid(material = np.where(threshold < GeomGrid._minimal_surface[surface](x,y,z),materials[1],materials[0]),
size = size,
comments = util.execution_stamp('GeomGrid','from_minimal_surface'),
)
2020-09-30 10:40:15 +05:30
def save(self,
fname: Union[str, Path],
compress: bool = True):
"""
2023-02-21 20:57:06 +05:30
Save as VTK ImageData file.
Parameters
----------
2020-11-20 00:56:15 +05:30
fname : str or pathlib.Path
2023-02-21 20:57:06 +05:30
Filename to write.
Valid extension is .vti, which will be appended if not given.
compress : bool, optional
Compress with zlib algorithm. Defaults to True.
"""
2022-02-21 16:47:00 +05:30
v = VTK.from_image_data(self.cells,self.size,self.origin)\
2022-05-12 04:24:03 +05:30
.set('material',self.material.flatten(order='F'))
for label,data in self.initial_conditions.items():
2022-05-12 04:24:03 +05:30
v = v.set(label,data.flatten(order='F'))
2022-03-09 20:05:36 +05:30
v.comments = self.comments
2022-01-13 01:04:29 +05:30
v.save(fname,parallel=False,compress=compress)
def save_ASCII(self,
fname: Union[str, TextIO]):
2019-11-23 01:22:36 +05:30
"""
2020-12-04 02:28:24 +05:30
Save as geom file.
2019-11-23 01:22:36 +05:30
2020-11-14 22:24:47 +05:30
Storing geometry files in ASCII format is deprecated.
This function will be removed in a future version of DAMASK.
2019-11-23 01:22:36 +05:30
Parameters
----------
fname : str or file handle
Geometry file to write with extension '.geom'.
2020-09-18 20:02:08 +05:30
compress : bool, optional
2023-02-21 20:57:06 +05:30
Compress geometry using 'x of y' and 'a to b'.
2019-11-23 01:22:36 +05:30
"""
2022-01-05 18:36:06 +05:30
warnings.warn('Support for ASCII-based geom format will be removed in DAMASK 3.0.0', DeprecationWarning,2)
header = [f'{len(self.comments)+4} header'] + self.comments \
2020-12-04 02:28:24 +05:30
+ ['grid a {} b {} c {}'.format(*self.cells),
'size x {} y {} z {}'.format(*self.size),
'origin x {} y {} z {}'.format(*self.origin),
'homogenization 1',
]
format_string = '%g' if self.material.dtype in np.sctypes['float'] else \
'%{}i'.format(1+int(np.floor(np.log10(np.nanmax(self.material)))))
np.savetxt(fname,
2020-12-04 02:28:24 +05:30
self.material.reshape([self.cells[0],np.prod(self.cells[1:])],order='F').T,
header='\n'.join(header), fmt=format_string, comments='')
2022-02-21 15:49:53 +05:30
def show(self,
colormap: Union[Colormap, str] = 'cividis') -> None:
2022-02-21 15:49:53 +05:30
"""
Show on screen.
Parameters
----------
colormap : damask.Colormap or str, optional
Colormap for visualization of material IDs. Defaults to 'cividis'.
2022-02-21 15:49:53 +05:30
"""
VTK.from_image_data(self.cells,self.size,self.origin) \
2022-05-12 04:24:03 +05:30
.set('material',self.material.flatten('F'),) \
.show('material',colormap)
def canvas(self,
2022-11-23 02:56:15 +05:30
cells: Optional[IntSequence] = None,
offset: Optional[IntSequence] = None,
2023-11-28 03:11:28 +05:30
fill: Optional[int] = None) -> 'GeomGrid':
2020-08-08 23:44:30 +05:30
"""
Crop or enlarge/pad grid.
2020-08-08 23:44:30 +05:30
Parameters
----------
cells : sequence of int, len (3), optional
2023-02-21 20:57:06 +05:30
Cell counts along x,y,z direction.
offset : sequence of int, len (3), optional
Offset (measured in cells) from old to new grid.
Defaults to [0,0,0].
2020-08-09 00:26:17 +05:30
fill : int, optional
Material ID to fill the background.
2023-02-21 20:57:06 +05:30
Defaults to material.max()+1.
2020-08-08 23:44:30 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
2021-07-02 09:46:12 +05:30
Examples
--------
Remove lower 1/2 of the microstructure in z-direction.
2021-07-02 09:46:12 +05:30
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> g = damask.GeomGrid(np.zeros([32]*3,int),np.ones(3)*1e-3)
>>> g.canvas([32,32,16],[0,0,16])
2023-02-21 20:57:06 +05:30
cells: 32 × 32 × 16
size: 0.001 × 0.001 × 0.0005
origin: 0.0 0.0 0.0005 m
# materials: 1
2021-07-02 09:46:12 +05:30
2020-08-08 23:44:30 +05:30
"""
2022-04-22 23:57:03 +05:30
offset_ = np.array(offset,np.int64) if offset is not None else np.zeros(3,np.int64)
cells_ = np.array(cells,np.int64) if cells is not None else self.cells
2020-08-08 23:44:30 +05:30
canvas = np.full(cells_,np.nanmax(self.material) + 1 if fill is None else fill,self.material.dtype)
2020-08-08 23:44:30 +05:30
LL = np.clip( offset_, 0,np.minimum(self.cells, cells_+offset_))
UR = np.clip( offset_+cells_, 0,np.minimum(self.cells, cells_+offset_))
ll = np.clip(-offset_, 0,np.minimum( cells_,self.cells-offset_))
ur = np.clip(-offset_+self.cells,0,np.minimum( cells_,self.cells-offset_))
2020-08-08 23:44:30 +05:30
canvas[ll[0]:ur[0],ll[1]:ur[1],ll[2]:ur[2]] = self.material[LL[0]:UR[0],LL[1]:UR[1],LL[2]:UR[2]]
2020-08-08 23:44:30 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = canvas,
size = self.size/self.cells*np.asarray(canvas.shape),
origin = self.origin+offset_*self.size/self.cells,
comments = self.comments+[util.execution_stamp('GeomGrid','canvas')],
)
2020-08-08 23:44:30 +05:30
def mirror(self,
directions: Sequence[str],
2023-11-28 03:11:28 +05:30
reflect: bool = False) -> 'GeomGrid':
"""
2020-12-04 11:42:18 +05:30
Mirror grid along given directions.
2019-11-24 13:22:46 +05:30
Parameters
----------
2022-01-30 03:46:57 +05:30
directions : (sequence of) {'x', 'y', 'z'}
2020-12-04 11:42:18 +05:30
Direction(s) along which the grid is mirrored.
reflect : bool, optional
2020-08-25 12:04:04 +05:30
Reflect (include) outermost layers. Defaults to False.
2019-11-24 13:22:46 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
2021-07-02 09:46:12 +05:30
Examples
--------
2023-02-21 20:57:06 +05:30
Mirror along y-direction.
2021-07-02 09:46:12 +05:30
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> (g := damask.GeomGrid(np.arange(4*5*6).reshape([4,5,6]),np.ones(3)))
2023-02-21 20:57:06 +05:30
cells: 4 × 5 × 6
size: 1.0 × 1.0 × 1.0
2022-02-17 11:43:39 +05:30
origin: 0.0 0.0 0.0 m
2023-02-21 20:57:06 +05:30
# materials: 120
>>> g.mirror('y')
cells: 4 × 8 × 6
size: 1.0 × 1.6 × 1.0
origin: 0.0 0.0 0.0 m
# materials: 120
Reflect along x- and y-direction.
>>> g.mirror('xy',reflect=True)
cells: 8 × 10 × 6
size: 2.0 × 2.0 × 1.0
origin: 0.0 0.0 0.0 m
# materials: 120
Independence of mirroring order.
>>> g.mirror('xy') == g.mirror(['y','x'])
True
2021-07-02 09:46:12 +05:30
"""
if not set(directions).issubset(valid := ['x', 'y', 'z']):
raise ValueError(f'invalid direction "{set(directions).difference(valid)}" specified')
2021-12-06 18:52:52 +05:30
limits: Sequence[Optional[int]] = [None,None] if reflect else [-2,0]
mat = self.material.copy()
if 'x' in directions:
mat = np.concatenate([mat,mat[limits[0]:limits[1]:-1,:,:]],0)
if 'y' in directions:
mat = np.concatenate([mat,mat[:,limits[0]:limits[1]:-1,:]],1)
if 'z' in directions:
mat = np.concatenate([mat,mat[:,:,limits[0]:limits[1]:-1]],2)
2020-03-21 15:37:21 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = mat,
size = self.size/self.cells*np.asarray(mat.shape),
origin = self.origin,
comments = self.comments+[util.execution_stamp('GeomGrid','mirror')],
)
def flip(self,
2023-11-28 03:11:28 +05:30
directions: Sequence[str]) -> 'GeomGrid':
"""
2020-12-04 11:42:18 +05:30
Flip grid along given directions.
Parameters
----------
2022-01-30 03:46:57 +05:30
directions : (sequence of) {'x', 'y', 'z'}
2020-12-04 11:42:18 +05:30
Direction(s) along which the grid is flipped.
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
2023-02-21 20:57:06 +05:30
Examples
--------
Invariance of flipping order.
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> (g := damask.GeomGrid(np.arange(4*5*6).reshape([4,5,6]),np.ones(3)))
2023-02-21 20:57:06 +05:30
cells: 4 × 5 × 6
size: 1.0 × 1.0 × 1.0
origin: 0.0 0.0 0.0 m
# materials: 120
>>> g.flip('xyz') == g.flip(['x','z','y'])
True
Invariance of flipping a (fully) mirrored grid.
>>> g.mirror('x',True) == g.mirror('x',True).flip('x')
True
"""
if not set(directions).issubset(valid := ['x', 'y', 'z']):
raise ValueError(f'invalid direction "{set(directions).difference(valid)}" specified')
mat = np.flip(self.material, [valid.index(d) for d in directions if d in valid])
2023-11-28 03:11:28 +05:30
return GeomGrid(material = mat,
size = self.size,
origin = self.origin,
comments = self.comments+[util.execution_stamp('GeomGrid','flip')],
)
def rotate(self,
R: Rotation,
2023-11-28 03:11:28 +05:30
fill: Optional[int] = None) -> 'GeomGrid':
"""
2023-02-21 20:57:06 +05:30
Rotate grid (possibly extending its bounding box).
Parameters
----------
R : damask.Rotation
Rotation to apply to the grid.
fill : int, optional
Material ID to fill enlarged bounding box.
2023-02-21 20:57:06 +05:30
Defaults to material.max()+1.
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
Updated grid-based geometry.
2023-02-21 20:57:06 +05:30
Examples
--------
Rotation by 180° (π) is equivalent to twice flipping.
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> (g := damask.GeomGrid(np.arange(4*5*6).reshape([4,5,6]),np.ones(3)))
2023-02-21 20:57:06 +05:30
cells: 4 × 5 × 6
size: 1.0 × 1.0 × 1.0
origin: 0.0 0.0 0.0 m
# materials: 120
>>> g.rotate(damask.Rotation.from_axis_angle([0,0,1,180],degrees=True)) == g.flip('xy')
True
"""
material = self.material
# These rotations are always applied in the reference coordinate system, i.e. (z,x,z) not (z,x',z'')
# see https://www.cs.utexas.edu/~theshark/courses/cs354/lectures/cs354-14.pdf
for angle,axes in zip(R.as_Euler_angles(degrees=True)[::-1], [(0,1),(1,2),(0,1)]):
material_temp = ndimage.rotate(material,angle,axes,order=0,prefilter=False,
output=self.material.dtype,
cval=np.nanmax(self.material) + 1 if fill is None else fill)
# avoid scipy interpolation errors for rotations close to multiples of 90°
material = material_temp if np.prod(material_temp.shape) != np.prod(material.shape) else \
np.rot90(material,k=np.rint(angle/90.).astype(np.int64),axes=axes)
origin = self.origin-(np.asarray(material.shape)-self.cells)*.5 * self.size/self.cells
2023-11-28 03:11:28 +05:30
return GeomGrid(material = material,
size = self.size/self.cells*np.asarray(material.shape),
origin = origin,
comments = self.comments+[util.execution_stamp('GeomGrid','rotate')],
)
def scale(self,
2023-11-28 03:11:28 +05:30
cells: IntSequence) -> 'GeomGrid':
2019-11-24 19:43:26 +05:30
"""
2023-02-21 20:57:06 +05:30
Scale grid to new cell counts.
2019-11-24 19:43:26 +05:30
Parameters
----------
2022-01-13 01:04:29 +05:30
cells : sequence of int, len (3)
2023-02-21 20:57:06 +05:30
Cell counts along x,y,z direction.
2019-11-24 19:43:26 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
2021-07-02 09:46:12 +05:30
Examples
--------
2023-02-21 20:57:06 +05:30
Double grid resolution.
2021-07-02 09:46:12 +05:30
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> (g := damask.GeomGrid(np.zeros([32]*3,int),np.ones(3)*1e-4))
2022-11-09 00:22:08 +05:30
cells: 32 × 32 × 32
size: 0.0001 × 0.0001 × 0.0001
origin: 0.0 0.0 0.0 m
# materials: 1
2021-07-02 11:18:01 +05:30
>>> g.scale(g.cells*2)
2023-02-21 20:57:06 +05:30
cells : 64 × 64 × 64
size : 0.0001 × 0.0001 × 0.0001
2022-02-17 11:43:39 +05:30
origin: 0.0 0.0 0.0 m
2021-07-02 09:46:12 +05:30
# materials: 1
2019-11-24 19:43:26 +05:30
"""
2022-11-09 00:22:08 +05:30
orig = tuple(map(np.linspace,self.origin + self.size/self.cells*.5,
self.origin + self.size - self.size/self.cells*.5,self.cells))
2022-11-14 17:12:13 +05:30
interpolator = partial(interpolate.RegularGridInterpolator,
points=orig,method='nearest',bounds_error=False,fill_value=None)
2022-11-09 00:22:08 +05:30
new = grid_filters.coordinates0_point(cells,self.size,self.origin)
2023-11-28 03:11:28 +05:30
return GeomGrid(material = interpolator(values=self.material)(new).astype(int),
size = self.size,
origin = self.origin,
initial_conditions = {k: interpolator(values=v)(new)
for k,v in self.initial_conditions.items()},
comments = self.comments+[util.execution_stamp('GeomGrid','scale')],
)
2019-11-24 19:43:26 +05:30
2022-11-09 00:22:08 +05:30
def assemble(self,
2023-11-28 03:11:28 +05:30
idx: np.ndarray) -> 'GeomGrid':
2022-11-09 00:22:08 +05:30
"""
Assemble new grid from index map.
Parameters
----------
idx : numpy.ndarray of int, shape (:,:,:) or (:,:,:,3)
2023-11-28 03:11:28 +05:30
GeomGrid of flat indices or coordinate indices.
2022-11-09 00:22:08 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2022-11-09 00:22:08 +05:30
Updated grid-based geometry.
Cell count of resulting grid matches shape of index map.
"""
cells = idx.shape[:3]
flat = (idx if len(idx.shape)==3 else grid_filters.ravel_index(idx)).flatten(order='F')
ic = {k: v.flatten(order='F')[flat].reshape(cells,order='F') for k,v in self.initial_conditions.items()}
2023-11-28 03:11:28 +05:30
return GeomGrid(material = self.material.flatten(order='F')[flat].reshape(cells,order='F'),
size = self.size,
origin = self.origin,
initial_conditions = ic,
comments = self.comments+[util.execution_stamp('GeomGrid','assemble')],
)
2022-11-09 00:22:08 +05:30
2023-11-28 03:11:28 +05:30
def renumber(self) -> 'GeomGrid':
"""
Renumber sorted material indices as 0,...,N-1.
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
Updated grid-based geometry.
"""
_,renumbered = np.unique(self.material,return_inverse=True)
2023-11-28 03:11:28 +05:30
return GeomGrid(material = renumbered.reshape(self.cells),
size = self.size,
origin = self.origin,
initial_conditions = self.initial_conditions,
comments = self.comments+[util.execution_stamp('GeomGrid','renumber')],
)
def substitute(self,
from_material: Union[int,IntSequence],
2023-11-28 03:11:28 +05:30
to_material: Union[int,IntSequence]) -> 'GeomGrid':
"""
Substitute material indices.
Parameters
----------
from_material : (sequence of) int
Material indices to be substituted.
to_material : (sequence of) int
New material indices.
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
Updated grid-based geometry.
"""
material = self.material.copy()
for f,t in zip(from_material if isinstance(from_material,(Sequence,np.ndarray)) else [from_material],
to_material if isinstance(to_material,(Sequence,np.ndarray)) else [to_material]): # ToDo Python 3.10 has strict mode for zip
material[self.material==f] = t
2023-11-28 03:11:28 +05:30
return GeomGrid(material = material,
size = self.size,
origin = self.origin,
initial_conditions = self.initial_conditions,
comments = self.comments+[util.execution_stamp('GeomGrid','substitute')],
)
2023-11-28 03:11:28 +05:30
def sort(self) -> 'GeomGrid':
"""
2023-02-21 20:57:06 +05:30
Sort material indices such that min(material ID) is located at (0,0,0).
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
Updated grid-based geometry.
"""
a = self.material.flatten(order='F')
from_ma = pd.unique(a)
sort_idx = np.argsort(from_ma)
ma = np.unique(a)[sort_idx][np.searchsorted(from_ma,a,sorter = sort_idx)]
2023-11-28 03:11:28 +05:30
return GeomGrid(material = ma.reshape(self.cells,order='F'),
size = self.size,
origin = self.origin,
initial_conditions = self.initial_conditions,
comments = self.comments+[util.execution_stamp('GeomGrid','sort')],
)
2022-01-13 01:04:29 +05:30
def clean(self,
distance: float = np.sqrt(3),
selection: Optional[IntSequence] = None,
invert_selection: bool = False,
periodic: bool = True,
2023-11-28 03:11:28 +05:30
rng_seed: Optional[NumpyRngSeed] = None) -> 'GeomGrid':
"""
Smooth grid by selecting most frequent material ID within given stencil at each location.
2019-11-24 13:22:46 +05:30
Parameters
----------
distance : float, optional
Voxel distance checked for presence of other materials.
Defaults to sqrt(3).
selection : (sequence of) int, optional
Material IDs to consider. Defaults to all.
invert_selection : bool, optional
Consider all material IDs except those in selection. Defaults to False.
2022-01-13 03:43:38 +05:30
periodic : bool, optional
2020-12-04 11:42:18 +05:30
Assume grid to be periodic. Defaults to True.
rng_seed : {None, int, array_like[ints], SeedSequence, BitGenerator, Generator}, optional
A seed to initialize the BitGenerator. Defaults to None.
If None, then fresh, unpredictable entropy will be pulled from the OS.
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
Notes
-----
If multiple material IDs are most frequent within a stencil, a random choice is taken.
"""
def most_frequent(stencil: np.ndarray,
selection: Union[None,np.ndarray],
rng: np.random.Generator):
me = stencil[stencil.size//2]
if selection is None or me in selection:
unique, counts = np.unique(stencil,return_counts=True)
return rng.choice(unique[counts==np.max(counts)])
else:
return me
rng = np.random.default_rng(rng_seed)
d = np.floor(distance).astype(np.int64)
ext = np.linspace(-d,d,1+2*d,dtype=float),
xx,yy,zz = np.meshgrid(ext,ext,ext)
footprint = xx**2+yy**2+zz**2 <= distance**2+distance*1e-8
selection_ = None if selection is None else \
np.setdiff1d(self.material,selection) if invert_selection else \
np.intersect1d(self.material,selection)
2022-11-09 00:22:08 +05:30
material = ndimage.generic_filter(
self.material,
most_frequent,
footprint=footprint,
mode='wrap' if periodic else 'nearest',
extra_keywords=dict(selection=selection_,rng=rng),
).astype(self.material.dtype)
2023-11-28 03:11:28 +05:30
return GeomGrid(material = material,
size = self.size,
origin = self.origin,
initial_conditions = self.initial_conditions,
comments = self.comments+[util.execution_stamp('GeomGrid','clean')],
)
2019-11-24 22:51:05 +05:30
def add_primitive(self,
dimension: Union[FloatSequence, IntSequence],
center: Union[FloatSequence, IntSequence],
exponent: Union[FloatSequence, float],
2022-11-23 02:56:15 +05:30
fill: Optional[int] = None,
R: Rotation = Rotation(),
inverse: bool = False,
2023-11-28 03:11:28 +05:30
periodic: bool = True) -> 'GeomGrid':
2020-05-30 21:01:50 +05:30
"""
Insert a primitive geometric object at a given position.
2020-05-30 21:01:50 +05:30
Parameters
----------
dimension : sequence of int or float, len (3)
Dimension (diameter/side length) of the primitive.
If given as integers, cell centers are addressed.
If given as floats, physical coordinates are addressed.
center : sequence of int or float, len (3)
Center of the primitive.
If given as integers, cell centers are addressed.
If given as floats, physical coordinates are addressed.
exponent : (sequence of) float, len (3)
Exponents for the three axes.
0 gives octahedron (ǀxǀ^(2^0) + ǀyǀ^(2^0) + ǀzǀ^(2^0) < 1)
1 gives sphere (ǀxǀ^(2^1) + ǀyǀ^(2^1) + ǀzǀ^(2^1) < 1)
2022-01-13 01:04:29 +05:30
fill : int, optional
Fill value for primitive. Defaults to material.max()+1.
R : damask.Rotation, optional
2023-02-21 20:57:06 +05:30
Rotation of the primitive. Defaults to no rotation.
inverse : bool, optional
Retain original materials within primitive and fill outside.
Defaults to False.
periodic : bool, optional
Assume grid to be periodic. Defaults to True.
2020-05-30 21:01:50 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
2021-07-02 09:46:12 +05:30
Examples
--------
Add a sphere at the center.
2021-07-02 09:46:12 +05:30
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> g = damask.GeomGrid(np.zeros([64]*3,int), np.ones(3)*1e-4)
>>> g.add_primitive(np.ones(3)*5e-5,np.ones(3)*5e-5,1)
2023-02-21 20:57:06 +05:30
cells : 64 × 64 × 64
size : 0.0001 × 0.0001 × 0.0001
origin: 0.0 0.0 0.0 m
# materials: 2
2020-05-30 21:01:50 +05:30
Add a cube at the origin.
2020-05-30 21:01:50 +05:30
>>> import numpy as np
>>> import damask
2023-11-28 03:11:28 +05:30
>>> g = damask.GeomGrid(np.zeros([64]*3,int), np.ones(3)*1e-4)
>>> g.add_primitive(np.ones(3,int)*32,np.zeros(3),np.inf)
2023-02-21 20:57:06 +05:30
cells : 64 × 64 × 64
size : 0.0001 × 0.0001 × 0.0001
origin: 0.0 0.0 0.0 m
# materials: 2
2020-05-30 21:01:50 +05:30
"""
# radius and center
r = np.array(dimension)/2.0*self.size/self.cells if np.array(dimension).dtype in np.sctypes['int'] else \
np.array(dimension)/2.0
c = (np.array(center) + .5)*self.size/self.cells if np.array(center).dtype in np.sctypes['int'] else \
(np.array(center) - self.origin)
2020-08-08 23:12:34 +05:30
coords = grid_filters.coordinates0_point(self.cells,self.size,
-(0.5*(self.size + (self.size/self.cells
if np.array(center).dtype in np.sctypes['int'] else
0)) if periodic else c))
coords_rot = R.broadcast_to(tuple(self.cells))@coords
with np.errstate(all='ignore'):
2022-08-29 17:14:50 +05:30
mask = np.sum(np.power(np.abs(coords_rot)/r,2.0**np.array(exponent)),axis=-1) > 1.0
if periodic: # translate back to center
mask = np.roll(mask,((c/self.size-0.5)*self.cells).round().astype(np.int64),(0,1,2))
2023-11-28 03:11:28 +05:30
return GeomGrid(material = np.where(np.logical_not(mask) if inverse else mask,
self.material,
np.nanmax(self.material)+1 if fill is None else fill),
2023-11-28 03:11:28 +05:30
size = self.size,
origin = self.origin,
initial_conditions = self.initial_conditions,
comments = self.comments+[util.execution_stamp('GeomGrid','add_primitive')],
)
2021-12-06 18:52:52 +05:30
def vicinity_offset(self,
distance: float = np.sqrt(3),
2022-11-23 02:56:15 +05:30
offset: Optional[int] = None,
selection: Optional[IntSequence] = None,
invert_selection: bool = False,
2023-11-28 03:11:28 +05:30
periodic: bool = True) -> 'GeomGrid':
2020-08-08 23:12:34 +05:30
"""
Offset material ID of points in the vicinity of selected (or just other) material IDs.
2020-08-08 23:12:34 +05:30
2022-03-09 20:05:36 +05:30
Trigger points are variations in material ID, i.e. grain/phase
boundaries or explicitly given material IDs.
2020-08-08 23:12:34 +05:30
Parameters
----------
distance : float, optional
Voxel distance checked for presence of other materials.
Defaults to sqrt(3).
2020-08-08 23:12:34 +05:30
offset : int, optional
Offset (positive or negative) to tag material IDs.
Defaults to material.max()+1.
selection : (sequence of) int, optional
Material IDs that trigger an offset.
Defaults to any other than own material ID.
invert_selection : bool, optional
Consider all material IDs except those in selection.
Defaults to False.
2022-01-13 03:43:38 +05:30
periodic : bool, optional
2020-12-04 11:42:18 +05:30
Assume grid to be periodic. Defaults to True.
2020-08-08 23:12:34 +05:30
Returns
-------
2023-11-28 03:11:28 +05:30
updated : damask.GeomGrid
2021-04-24 18:17:52 +05:30
Updated grid-based geometry.
2020-08-08 23:12:34 +05:30
"""
@numba_njit_wrapper()
def tainted_neighborhood(stencil: np.ndarray,
selection: Optional[np.ndarray] = None):
me = stencil[stencil.size//2]
if selection is None:
return np.any(stencil != me)
elif not len(selection)==0:
for stencil_item in stencil:
for selection_item in selection:
if stencil_item==selection_item and selection_item!=me:
return True
return False
d = np.floor(distance).astype(np.int64)
ext = np.linspace(-d,d,1+2*d,dtype=float),
xx,yy,zz = np.meshgrid(ext,ext,ext)
footprint = xx**2+yy**2+zz**2 <= distance**2+distance*1e-8
offset_ = np.nanmax(self.material)+1 if offset is None else offset
selection_ = None if selection is None else \
np.setdiff1d(self.material,selection) if invert_selection else \
np.intersect1d(self.material,selection)
2022-11-09 00:22:08 +05:30
mask = ndimage.generic_filter(self.material,
tainted_neighborhood,
footprint=footprint,
mode='wrap' if periodic else 'nearest',
extra_keywords=dict(selection=selection_),
)
2020-08-08 23:12:34 +05:30
2023-11-28 03:11:28 +05:30
return GeomGrid(material = np.where(mask, self.material + offset_,self.material),
size = self.size,
origin = self.origin,
initial_conditions = self.initial_conditions,
comments = self.comments+[util.execution_stamp('GeomGrid','vicinity_offset')],
)
2020-11-28 00:46:06 +05:30
def get_grain_boundaries(self,
periodic: bool = True,
directions: Sequence[str] = 'xyz') -> VTK:
2020-11-18 16:55:08 +05:30
"""
2020-11-28 00:46:06 +05:30
Create VTK unstructured grid containing grain boundaries.
2020-11-18 16:44:12 +05:30
Parameters
----------
2022-01-13 03:43:38 +05:30
periodic : bool, optional
2020-12-04 11:42:18 +05:30
Assume grid to be periodic. Defaults to True.
2022-01-30 03:46:57 +05:30
directions : (sequence of) {'x', 'y', 'z'}, optional
2020-12-04 11:42:18 +05:30
Direction(s) along which the boundaries are determined.
2022-01-30 03:46:57 +05:30
Defaults to 'xyz'.
2020-11-18 16:44:12 +05:30
2021-04-25 11:17:00 +05:30
Returns
-------
grain_boundaries : damask.VTK
VTK-based geometry of grain boundary network.
2020-11-18 16:44:12 +05:30
"""
if not set(directions).issubset(valid := ['x', 'y', 'z']):
raise ValueError(f'invalid direction "{set(directions).difference(valid)}" specified')
2020-11-19 00:40:04 +05:30
2020-12-04 02:28:24 +05:30
o = [[0, self.cells[0]+1, np.prod(self.cells[:2]+1)+self.cells[0]+1, np.prod(self.cells[:2]+1)],
[0, np.prod(self.cells[:2]+1), np.prod(self.cells[:2]+1)+1, 1],
[0, 1, self.cells[0]+1+1, self.cells[0]+1]] # offset for connectivity
2020-11-28 00:46:06 +05:30
connectivity = []
for i,d in enumerate(['x','y','z']):
if d not in directions: continue
2020-11-28 00:46:06 +05:30
mask = self.material != np.roll(self.material,1,i)
for j in [0,1,2]:
mask = np.concatenate((mask,np.take(mask,[0],j)*(i==j)),j)
if i == 0 and not periodic: mask[0,:,:] = mask[-1,:,:] = False
if i == 1 and not periodic: mask[:,0,:] = mask[:,-1,:] = False
if i == 2 and not periodic: mask[:,:,0] = mask[:,:,-1] = False
2020-11-28 00:46:06 +05:30
base_nodes = np.argwhere(mask.flatten(order='F')).reshape(-1,1)
connectivity.append(np.block([base_nodes + o[i][k] for k in range(4)]))
coords = grid_filters.coordinates0_node(self.cells,self.size,self.origin).reshape(-1,3,order='F')
return VTK.from_unstructured_grid(coords,np.vstack(connectivity),'QUAD')