Files
tono/backend/nodes/trimmed_mean.py
2026-04-03 23:11:52 -07:00

55 lines
1.7 KiB
Python

"""Trimmed mean filter — mean filter excluding extreme percentiles."""
from __future__ import annotations
import numpy as np
from backend.node_registry import register_node
from backend.data_types import DataField
@register_node(display_name="Trimmed Mean")
class TrimmedMean:
@classmethod
def INPUT_TYPES(cls):
return {
"required": {
"field": ("DATA_FIELD",),
"radius": ("INT", {"default": 3, "min": 1, "max": 50, "step": 1}),
"trim_fraction": ("FLOAT", {"default": 0.1, "min": 0.0, "max": 0.45, "step": 0.01}),
}
}
OUTPUTS = (
('DATA_FIELD', 'filtered'),
)
FUNCTION = "process"
DESCRIPTION = (
"Apply a local mean filter that excludes the lowest and highest "
"fraction of values in each window. More robust than Gaussian for "
"data with outlier spikes. trim_fraction=0 is a plain mean; "
"trim_fraction=0.5 approaches the median. "
)
def process(self, field: DataField, radius: int, trim_fraction: float) -> tuple:
data = np.asarray(field.data, dtype=np.float64)
yres, xres = data.shape
result = np.zeros_like(data)
padded = np.pad(data, radius, mode='edge')
for iy in range(yres):
for ix in range(xres):
window = padded[iy:iy + 2 * radius + 1, ix:ix + 2 * radius + 1].ravel()
n = len(window)
k = int(n * trim_fraction)
if k > 0:
sorted_w = np.sort(window)
trimmed = sorted_w[k:n - k]
else:
trimmed = window
result[iy, ix] = trimmed.mean()
return (field.replace(data=result),)