Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add keypoints to 3d #2217

Merged
merged 9 commits into from
Dec 24, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 8 additions & 8 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -311,14 +311,14 @@ Where:
- Volume: 3D array of shape (D, H, W) or (D, H, W, C) where D is depth, H is height, W is width, and C is number of channels (optional)
- Mask3D: Binary or multi-class 3D mask of shape (D, H, W) where each slice represents segmentation for the corresponding volume slice

| Transform | Volume | Mask3D |
| ------------------------------------------------------------------------------ | :----: | :----: |
| [CenterCrop3D](https://explore.albumentations.ai/transform/CenterCrop3D) | ✓ | ✓ |
| [CoarseDropout3D](https://explore.albumentations.ai/transform/CoarseDropout3D) | ✓ | ✓ |
| [CubicSymmetry](https://explore.albumentations.ai/transform/CubicSymmetry) | ✓ | ✓ |
| [Pad3D](https://explore.albumentations.ai/transform/Pad3D) | ✓ | ✓ |
| [PadIfNeeded3D](https://explore.albumentations.ai/transform/PadIfNeeded3D) | ✓ | ✓ |
| [RandomCrop3D](https://explore.albumentations.ai/transform/RandomCrop3D) | ✓ | ✓ |
| Transform | Volume | Mask3D | Keypoints |
| ------------------------------------------------------------------------------ | :----: | :----: | :-------: |
| [CenterCrop3D](https://explore.albumentations.ai/transform/CenterCrop3D) | ✓ | ✓ | ✓ |
| [CoarseDropout3D](https://explore.albumentations.ai/transform/CoarseDropout3D) | ✓ | ✓ | ✓ |
| [CubicSymmetry](https://explore.albumentations.ai/transform/CubicSymmetry) | ✓ | ✓ | ✓ |
| [Pad3D](https://explore.albumentations.ai/transform/Pad3D) | ✓ | ✓ | ✓ |
| [PadIfNeeded3D](https://explore.albumentations.ai/transform/PadIfNeeded3D) | ✓ | ✓ | ✓ |
| [RandomCrop3D](https://explore.albumentations.ai/transform/RandomCrop3D) | ✓ | ✓ | ✓ |

## A few more examples of **augmentations**

Expand Down
176 changes: 166 additions & 10 deletions albumentations/augmentations/transforms3d/functional.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@

import numpy as np

from albumentations.augmentations.utils import handle_empty_array
from albumentations.core.types import NUM_VOLUME_DIMENSIONS, ColorType


Expand Down Expand Up @@ -48,30 +49,40 @@

def pad_3d_with_params(
volume: np.ndarray,
padding: tuple[int, int, int, int, int, int], # (d_front, d_back, h_top, h_bottom, w_left, w_right)
padding: tuple[int, int, int, int, int, int],
value: ColorType,
) -> np.ndarray:
"""Pad 3D image with given parameters.
"""Pad 3D volume with given parameters.

Args:
volume: Input volume with shape (depth, height, width) or (depth, height, width, channels)
padding: Padding values (d_front, d_back, h_top, h_bottom, w_left, w_right)
value: Padding value
padding: Padding values in format:
(depth_front, depth_back, height_top, height_bottom, width_left, width_right)
where:
- depth_front/back: padding at start/end of depth axis (z)
- height_top/bottom: padding at start/end of height axis (y)
- width_left/right: padding at start/end of width axis (x)
value: Value to fill the padding

Returns:
Padded image with same number of dimensions as input
Padded volume with same number of dimensions as input

Note:
The padding order matches the volume dimensions (depth, height, width).
For each dimension, the first value is padding at the start (smaller indices),
and the second value is padding at the end (larger indices).
"""
d_front, d_back, h_top, h_bottom, w_left, w_right = padding
depth_front, depth_back, height_top, height_bottom, width_left, width_right = padding

# Skip if no padding is needed
if d_front == d_back == h_top == h_bottom == w_left == w_right == 0:
if all(p == 0 for p in padding):
return volume

# Handle both 3D and 4D arrays
pad_width = [
(d_front, d_back), # depth padding
(h_top, h_bottom), # height padding
(w_left, w_right), # width padding
(depth_front, depth_back), # depth (z) padding
(height_top, height_bottom), # height (y) padding
(width_left, width_right), # width (x) padding
]

# Add channel padding if 4D array
Expand Down Expand Up @@ -153,3 +164,148 @@
return np.rot90(temp, rotation_index - 16, axes=(0, 2))
temp = np.rot90(working_cube, -1, axes=(0, 1))
return np.rot90(temp, rotation_index - 20, axes=(0, 2))


@handle_empty_array("keypoints")
def filter_keypoints_in_holes3d(keypoints: np.ndarray, holes: np.ndarray) -> np.ndarray:
"""Filter out keypoints that are inside any of the 3D holes.

Args:
keypoints (np.ndarray): Array of keypoints with shape (num_keypoints, 3+).
The first three columns are x, y, z coordinates.
holes (np.ndarray): Array of holes with shape (num_holes, 6).
Each hole is represented as [z1, y1, x1, z2, y2, x2].

Returns:
np.ndarray: Array of keypoints that are not inside any hole.
"""
if holes.size == 0:
return keypoints

# Broadcast keypoints and holes for vectorized comparison
# Convert keypoints from XYZ to ZYX for comparison with holes
kp_z = keypoints[:, 2][:, np.newaxis] # Shape: (num_keypoints, 1)
kp_y = keypoints[:, 1][:, np.newaxis] # Shape: (num_keypoints, 1)
kp_x = keypoints[:, 0][:, np.newaxis] # Shape: (num_keypoints, 1)

# Extract hole coordinates (in ZYX order)
hole_z1 = holes[:, 0] # Shape: (num_holes,)
hole_y1 = holes[:, 1]
hole_x1 = holes[:, 2]
hole_z2 = holes[:, 3]
hole_y2 = holes[:, 4]
hole_x2 = holes[:, 5]

# Check if each keypoint is inside each hole
inside_hole = (
(kp_z >= hole_z1)
& (kp_z < hole_z2)
& (kp_y >= hole_y1)
& (kp_y < hole_y2)
& (kp_x >= hole_x1)
& (kp_x < hole_x2)
)

# A keypoint is valid if it's not inside any hole
valid_keypoints = ~np.any(inside_hole, axis=1)

# Return filtered keypoints with same dtype as input
result = keypoints[valid_keypoints]
if len(result) == 0:
# Ensure empty result has correct shape and dtype
return np.array([], dtype=keypoints.dtype).reshape(0, keypoints.shape[1])
return result


def keypoints_rot90(
keypoints: np.ndarray,
k: int,
axes: tuple[int, int],
volume_shape: tuple[int, int, int],
) -> np.ndarray:
if k == 0 or len(keypoints) == 0:
ternaus marked this conversation as resolved.
Show resolved Hide resolved
return keypoints

# Normalize factor to range [0, 3]
k = ((k % 4) + 4) % 4

result = keypoints.copy()

# Get dimensions for the rotation axes
dims = [volume_shape[ax] for ax in axes]

# Get coordinates to rotate
coords1 = result[:, axes[0]].copy()
coords2 = result[:, axes[1]].copy()

# Apply rotation based on factor (counterclockwise)
if k == 1: # 90 degrees CCW
result[:, axes[0]] = (dims[1] - 1) - coords2
result[:, axes[1]] = coords1
elif k == 2: # 180 degrees
result[:, axes[0]] = (dims[0] - 1) - coords1
result[:, axes[1]] = (dims[1] - 1) - coords2
elif k == 3: # 270 degrees CCW
result[:, axes[0]] = coords2
result[:, axes[1]] = (dims[0] - 1) - coords1

return result


@handle_empty_array("keypoints")
def transform_cube_keypoints(
keypoints: np.ndarray,
index: int,
volume_shape: tuple[int, int, int],
) -> np.ndarray:
if not (0 <= index < 48):
raise ValueError("Index must be between 0 and 47")

Check warning on line 262 in albumentations/augmentations/transforms3d/functional.py

View check run for this annotation

Codecov / codecov/patch

albumentations/augmentations/transforms3d/functional.py#L262

Added line #L262 was not covered by tests

# Create working copy preserving all columns
working_points = keypoints.copy()

# Convert only XYZ coordinates to HWD, keeping other columns unchanged
xyz = working_points[:, :3] # Get first 3 columns (XYZ)
xyz = xyz[:, [2, 1, 0]] # XYZ -> HWD
working_points[:, :3] = xyz # Put back transformed coordinates

current_shape = volume_shape

# Handle reflection first (indices 24-47)
if index >= 24:
working_points[:, 2] = current_shape[2] - 1 - working_points[:, 2] # Reflect W axis

rotation_index = index % 24

# Apply the same rotation logic as transform_cube
if rotation_index < 4:
# First 4: rotate around axis 0
result = keypoints_rot90(working_points, k=rotation_index, axes=(1, 2), volume_shape=current_shape)
elif rotation_index < 8:
# Next 4: flip 180° about axis 1, then rotate around axis 0
temp = keypoints_rot90(working_points, k=2, axes=(0, 2), volume_shape=current_shape)
result = keypoints_rot90(temp, k=rotation_index - 4, axes=(1, 2), volume_shape=volume_shape)
elif rotation_index < 16:
if rotation_index < 12:
ternaus marked this conversation as resolved.
Show resolved Hide resolved
temp = keypoints_rot90(working_points, k=1, axes=(0, 2), volume_shape=current_shape)
temp_shape = (current_shape[2], current_shape[1], current_shape[0])
result = keypoints_rot90(temp, k=rotation_index - 8, axes=(0, 1), volume_shape=temp_shape)
else:
temp = keypoints_rot90(working_points, k=3, axes=(0, 2), volume_shape=current_shape)
temp_shape = (current_shape[2], current_shape[1], current_shape[0])
result = keypoints_rot90(temp, k=rotation_index - 12, axes=(0, 1), volume_shape=temp_shape)
elif rotation_index < 20:
temp = keypoints_rot90(working_points, k=1, axes=(0, 1), volume_shape=current_shape)
temp_shape = (current_shape[1], current_shape[0], current_shape[2])
result = keypoints_rot90(temp, k=rotation_index - 16, axes=(0, 2), volume_shape=temp_shape)
else:
temp = keypoints_rot90(working_points, k=3, axes=(0, 1), volume_shape=current_shape)
temp_shape = (current_shape[1], current_shape[0], current_shape[2])
result = keypoints_rot90(temp, k=rotation_index - 20, axes=(0, 2), volume_shape=temp_shape)

# Convert back from HWD to XYZ coordinates for first 3 columns only
xyz = result[:, :3]
xyz = xyz[:, [2, 1, 0]] # HWD -> XYZ
result[:, :3] = xyz

return result
Loading
Loading