R/PipeOpFDAFourier.R

#' @title Fast Fourier transform features
#' @name mlr_pipeops_fda.fourier
#'
#' @description
#' This `PipeOp` extracts Fourier coefficients from functional columns.
#' For more details, see [stats::fft()], which is called internally.
#' Only the one-sided spectrum is returned since the input is real-valued (Oppenheim and Schafer, 2010).
#'
#' @section Parameters:
#' The parameters are the parameters inherited from [`PipeOpTaskPreprocSimple`][mlr3pipelines::PipeOpTaskPreprocSimple],
#' as well as the following parameters:
#' * `type` :: `character(1)`\cr
#'   Which feature to extract from the Fourier coefficients. `"amplitude"` returns the magnitude.
#'   `"phase"` returns the phase shift in degrees (values in \[-180, 180\]). Initial value is `"phase"`.
#'
#' @references `r format_bib("oppenheim2010discrete")`
#'
#' @export
#' @examples
#' task = tsk("fuel")
#' po_fourier = po("fda.fourier")
#' task_fourier = po_fourier$train(list(task))[[1L]]
#' task_fourier$data()
PipeOpFDAFourier = R6Class("PipeOpFDAFourier",
  inherit = PipeOpTaskPreprocSimple,
  public = list(
    #' @description Initializes a new instance of this Class.
    #' @param id (`character(1)`)\cr
    #'   Identifier of resulting object, default is `"fda.fourier"`.
    #' @param param_vals (named `list()`)\cr
    #'   List of hyperparameter settings, overwriting the hyperparameter settings that would
    #'   otherwise be set during construction. Default `list()`.
    initialize = function(id = "fda.fourier", param_vals = list()) {
      param_set = ps(
        type = p_fct(levels = c("phase", "amplitude"), tags = c("train", "predict"))
      )
      param_set$set_values(type = "phase")

      super$initialize(
        id = id,
        param_set = param_set,
        param_vals = param_vals,
        packages = c("mlr3fda", "mlr3pipelines", "tf"),
        feature_types = "tfd_reg",
        tags = "fda"
      )
    }
  ),

  private = list(
    .transform_dt = function(dt, levels) {
      type = self$param_set$get_values()$type

      setcbindlist(imap(dt, function(x, nm) {
        values = tf::tf_evaluations(x)
        n = length(values[[1L]])
        n_onesided = n %/% 2L + 1L
        fft_coeff = map_dtr(values, function(v) {
          coeff = stats::fft(v)[seq_len(n_onesided)] / n
          res = switch(
            type,
            amplitude = {
              amp = Mod(coeff)
              # double interior positive frequencies (DC and Nyquist have no conjugate pair)
              double_end = if (n %% 2L == 0L) n_onesided - 1L else n_onesided
              if (double_end >= 2L) {
                amp[2L:double_end] = amp[2L:double_end] * 2
              }
              amp
            },
            phase = {
              mag = Mod(coeff)
              phase = Arg(coeff) * 180 / pi
              phase[mag < max(mag) * 1e-4] = 0
              phase
            }
          )
          as.data.table(t(res))
        })
        setnames(fft_coeff, sprintf("%s_fft_%s_%i", nm, type, seq_len(ncol(fft_coeff))))
      }))
    }
  )
)

#' @include zzz.R
register_po("fda.fourier", PipeOpFDAFourier)

Try the mlr3fda package in your browser

Any scripts or data that you put into this service are public.

mlr3fda documentation built on March 31, 2026, 1:06 a.m.