Skip to content

Commit

Permalink
ENH: Add volume averaging.
Browse files Browse the repository at this point in the history
  • Loading branch information
jwboth committed Jun 20, 2024
1 parent eed5d21 commit dad34ac
Show file tree
Hide file tree
Showing 2 changed files with 115 additions and 0 deletions.
1 change: 1 addition & 0 deletions src/darsia/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@
from darsia.restoration.binaryinpaint import *
from darsia.restoration.h1_regularization import *
from darsia.restoration.split_bregman_tvd import *
from darsia.restoration.averaging import *
from darsia.multi_image_analysis.translationanalysis import *
from darsia.multi_image_analysis.concentrationanalysis import *
from darsia.multi_image_analysis.model_calibration import *
Expand Down
114 changes: 114 additions & 0 deletions src/darsia/restoration/averaging.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
"""Module with tools for volume averaging."""

from typing import Optional, Union, overload

import numpy as np
import scipy.ndimage

import darsia


class REV:
"""Class defining an representative elementary volume."""

def __init__(self, size: Union[float, tuple[float]], img: darsia.Image) -> None:
"""Initialization of a REV.
Args:
size (float or tuple of float): size of REV in length units
img (Image): image, used for determining the number of voxels
"""
if isinstance(size, float):
size = [size] * img.coordinatesystem.dim
self.size: int = max(
[
img.coordinatesystem.num_voxels(size[i], axis=["x", "y", "z"][i])
for i in range(img.coordinatesystem.dim)
]
)
"""Size of the REV in number of voxels."""


class VolumeAveraging:

def __init__(self, rev: REV, mask: darsia.Image) -> None:
"""Constructor.
Args:
rev (REV): representative elementary volume
mask (Image): mask
"""
self.rev_size = rev.size
"""Size of the REV."""
self.mask = mask
"""Mask."""
self.mean_pore_volume = scipy.ndimage.uniform_filter(
self.mask.astype(float).img, size=self.rev_size
)
"""Mean pore volume."""
tol = 1e-12
self.zero_indices = np.where(self.mean_pore_volume < tol)
"""Zero indices in the mean pore volume."""

# User output
print(f"Number of zero indices: {len(self.zero_indices[0])}")

@overload # type: ignore [override]
def __call__(self, img: np.ndarray) -> np.ndarray: ...
@overload # type: ignore [override]
def __call__(self, img: darsia.Image) -> darsia.Image: ...
def __call__(
self, img: Union[np.ndarray, darsia.Image]
) -> Union[np.ndarray, darsia.Image]:
"""Application of volume averaging.
Args:
img (np.ndarray or Image): image
Returns:
np.ndarray or Image: volume averaged image (same type as input)
"""
if isinstance(img, np.ndarray):
return self._average_array(img)
elif isinstance(img, darsia.Image):
result = img.copy()
result.img = self._average_array(img.img)
return result

def _average_array(self, arr: np.ndarray) -> np.ndarray:
"""Application of volume averaging to numpy array.
Args:
arr (np.ndarray): array
Returns:
np.ndarray: volume averaged array
"""
masked_data = np.multiply(arr, self.mask.img)
mean_masked_data = scipy.ndimage.uniform_filter(masked_data, size=self.rev_size)
result = np.divide(mean_masked_data, self.mean_pore_volume)
result[self.zero_indices] = 0
return result


def volume_average(
img: darsia.Image, mask: darsia.Image, rev: Optional[float] = None
) -> darsia.Image:
"""Fast-access function for volume averaging.
Note: For repeated calls, it is recommended to create a VolumeAveraging object.
Args:
img (Image): image
mask (Image): mask
rev (float): size of the REV in length units
Returns:
Image: volume averaged image
"""
return VolumeAveraging(rev=REV(size=rev, img=img), mask=mask)(img)

0 comments on commit dad34ac

Please sign in to comment.