53 lines
1.6 KiB
Python
53 lines
1.6 KiB
Python
from __future__ import annotations
|
|
import numpy as np
|
|
from backend.node_registry import register_node
|
|
from backend.data_types import DataField
|
|
from backend.nodes.spectral_common import (
|
|
preprocess_spectral_data,
|
|
psdf_field_from_data,
|
|
spatial_frequency_field,
|
|
)
|
|
|
|
|
|
@register_node(display_name="FFT 2D")
|
|
class FFT2D:
|
|
@classmethod
|
|
def INPUT_TYPES(cls):
|
|
return {
|
|
"required": {
|
|
"field": ("DATA_FIELD",),
|
|
"windowing": (["hann", "hamming", "blackman", "none"],),
|
|
"level": (["mean", "plane", "none"],),
|
|
}
|
|
}
|
|
|
|
OUTPUTS = (
|
|
('DATA_FIELD', 'log_magnitude'),
|
|
('DATA_FIELD', 'magnitude'),
|
|
('DATA_FIELD', 'phase'),
|
|
('DATA_FIELD', 'psdf'),
|
|
)
|
|
FUNCTION = "process"
|
|
|
|
DESCRIPTION = (
|
|
"Compute the 2D FFT with optional windowing and mean/plane subtraction. "
|
|
"Outputs log magnitude, magnitude, phase, and PSDF as separate channels. "
|
|
)
|
|
|
|
KEYWORDS = ("fourier", "frequency", "spectrum", "psdf", "magnitude", "phase")
|
|
|
|
def process(self, field: DataField, windowing: str, level: str) -> tuple:
|
|
data = preprocess_spectral_data(field, level=level, windowing=windowing)
|
|
F = np.fft.fftshift(np.fft.fft2(data))
|
|
|
|
magnitude = np.abs(F)
|
|
log_magnitude = np.log1p(magnitude)
|
|
phase = np.angle(F)
|
|
|
|
return (
|
|
spatial_frequency_field(field, log_magnitude),
|
|
spatial_frequency_field(field, magnitude),
|
|
spatial_frequency_field(field, phase),
|
|
psdf_field_from_data(field, data),
|
|
)
|