48#include <unsupported/Eigen/FFT>
67constexpr double MORLET_PI = 3.14159265358979323846;
74int MorletTfr::nextPow2(
int n)
77 while (p < n) p <<= 1;
83VectorXcd MorletTfr::buildWavelet(
double dFreq,
double dSFreq,
double dNCycles,
int& halfLen)
86 const double sigma_t = dNCycles / (2.0 * MORLET_PI * dFreq);
89 halfLen =
static_cast<int>(std::round(4.0 * sigma_t * dSFreq));
91 const int nWav = 2 * halfLen + 1;
92 VectorXcd wavelet(nWav);
95 const double A = std::pow(sigma_t * std::sqrt(2.0 * MORLET_PI), -0.5);
97 for (
int i = 0; i < nWav; ++i) {
98 const double t =
static_cast<double>(i - halfLen) / dSFreq;
99 const double gauss = std::exp(-t * t / (2.0 * sigma_t * sigma_t));
100 const double phase = 2.0 * MORLET_PI * dFreq * t;
101 wavelet[i] = std::complex<double>(A * gauss * std::cos(phase),
102 A * gauss * std::sin(phase));
111 const RowVectorXd& vecFreqs,
114 const int nTimes =
static_cast<int>(vecData.cols());
115 const int nFreqs =
static_cast<int>(vecFreqs.cols());
118 result.
matPower.resize(nFreqs, nTimes);
123 Eigen::FFT<double> fft;
125 for (
int fi = 0; fi < nFreqs; ++fi) {
127 const VectorXcd wavelet = buildWavelet(vecFreqs[fi], dSFreq, dNCycles, halfLen);
129 const int nWav =
static_cast<int>(wavelet.size());
130 const int nConv = nextPow2(nTimes + nWav - 1);
133 VectorXd sigPad = VectorXd::Zero(nConv);
134 sigPad.head(nTimes) = vecData.transpose();
136 fft.fwd(sigSpec, sigPad);
139 VectorXd wavReal = VectorXd::Zero(nConv);
140 VectorXd wavImag = VectorXd::Zero(nConv);
141 for (
int k = 0; k < nWav; ++k) {
142 wavReal[k] = wavelet[k].real();
143 wavImag[k] = wavelet[k].imag();
145 VectorXcd wavSpecR, wavSpecI;
146 fft.fwd(wavSpecR, wavReal);
147 fft.fwd(wavSpecI, wavImag);
149 VectorXcd wavSpec = wavSpecR + std::complex<double>(0.0, 1.0) * wavSpecI;
152 VectorXcd product = sigSpec.array() * wavSpec.array();
154 fft.inv(conv, product);
157 for (
int t = 0; t < nTimes; ++t)
158 result.
matPower(fi, t) = std::norm(conv[t + halfLen]);
168 const RowVectorXd& vecFreqs,
170 const RowVectorXi& vecPicks)
172 std::vector<int> picks;
173 if (vecPicks.size() > 0) {
174 picks.reserve(
static_cast<std::size_t
>(vecPicks.size()));
175 for (
int i = 0; i < vecPicks.size(); ++i)
176 picks.push_back(vecPicks[i]);
178 picks.reserve(
static_cast<std::size_t
>(matData.rows()));
179 for (
int i = 0; i < static_cast<int>(matData.rows()); ++i)
183 QVector<MorletTfrResult> results;
184 results.reserve(
static_cast<int>(picks.size()));
186 results.append(
compute(matData.row(ch), dSFreq, vecFreqs, dNCycles));
MorletTfr class declaration — complex Morlet wavelet time-frequency representation.
Shared utilities (I/O helpers, spectral analysis, layout management, warp algorithms).
Result of a Morlet TFR computation for one channel.
Eigen::RowVectorXd vecFreqs
Centre frequencies in Hz, length n_freqs.
Eigen::MatrixXd matPower
n_freqs × n_times, instantaneous power (amplitude²)
static MorletTfrResult compute(const Eigen::RowVectorXd &vecData, double dSFreq, const Eigen::RowVectorXd &vecFreqs, double dNCycles=7.0)
static QVector< MorletTfrResult > computeMultiChannel(const Eigen::MatrixXd &matData, double dSFreq, const Eigen::RowVectorXd &vecFreqs, double dNCycles=7.0, const Eigen::RowVectorXi &vecPicks=Eigen::RowVectorXi())