77 if (epochs.isEmpty()) {
78 qWarning() <<
"Xdawn::fit: empty epoch list.";
83 goodIdx.reserve(epochs.size());
84 for (
int i = 0; i < epochs.size(); ++i) {
85 if (!epochs[i].bReject) {
90 if (goodIdx.isEmpty()) {
91 qWarning() <<
"Xdawn::fit: no non-rejected epochs available.";
95 const int nCh =
static_cast<int>(epochs[goodIdx[0]].epoch.rows());
96 const int nSamp =
static_cast<int>(epochs[goodIdx[0]].epoch.cols());
97 if (nCh == 0 || nSamp == 0) {
98 qWarning() <<
"Xdawn::fit: epoch matrices are empty.";
102 for (
int idx : goodIdx) {
103 if (epochs[idx].epoch.rows() != nCh || epochs[idx].epoch.cols() != nSamp) {
104 qWarning() <<
"Xdawn::fit: epoch dimension mismatch.";
109 nComponents = std::max(1, std::min(nComponents, nCh));
111 QHash<int, MatrixXd> classSums;
112 QHash<int, int> classCounts;
113 MatrixXd targetSum = MatrixXd::Zero(nCh, nSamp);
116 for (
int idx : goodIdx) {
118 if (!classSums.contains(ep.
event)) {
119 classSums.insert(ep.
event, MatrixXd::Zero(nCh, nSamp));
120 classCounts.insert(ep.
event, 0);
124 classCounts[ep.
event] += 1;
126 if (ep.
event == iTargetEvent) {
127 targetSum += ep.
epoch;
133 qWarning() <<
"Xdawn::fit: no target epochs found for event" << iTargetEvent;
139 MatrixXd noiseCov = MatrixXd::Zero(nCh, nCh);
140 MatrixXd dataCov = MatrixXd::Zero(nCh, nCh);
141 long long nNoiseSamples = 0;
142 long long nDataSamples = 0;
144 QHash<int, MatrixXd> classMeans;
145 for (
auto it = classSums.constBegin(); it != classSums.constEnd(); ++it) {
146 classMeans.insert(it.key(), it.value() /
static_cast<double>(classCounts.value(it.key())));
149 for (
int idx : goodIdx) {
151 const MatrixXd residual = ep.
epoch - classMeans.value(ep.
event);
154 noiseCov += residual * residual.transpose();
155 nDataSamples += nSamp;
156 nNoiseSamples += nSamp;
159 if (nNoiseSamples <= 0 || nDataSamples <= 0) {
160 qWarning() <<
"Xdawn::fit: failed to accumulate covariance samples.";
165 result.
matNoiseCov = noiseCov /
static_cast<double>(nNoiseSamples);
166 dataCov = dataCov /
static_cast<double>(nDataSamples);
168 const double traceNoise = result.
matNoiseCov.trace();
169 const double regValue = std::max(dReg, 0.0) * ((traceNoise > 0.0) ? traceNoise /
static_cast<double>(nCh) : 1.0);
171 regNoiseCov.diagonal().array() += regValue;
173 SelfAdjointEigenSolver<MatrixXd> noiseEig(regNoiseCov);
174 if (noiseEig.info() != Success) {
175 qWarning() <<
"Xdawn::fit: noise covariance eigendecomposition failed.";
179 VectorXd noiseVals = noiseEig.eigenvalues().cwiseMax(1e-12);
180 MatrixXd noiseVecs = noiseEig.eigenvectors();
181 MatrixXd invSqrtNoise = noiseVecs * noiseVals.cwiseInverse().cwiseSqrt().asDiagonal() * noiseVecs.transpose();
183 MatrixXd whitenedSignal = invSqrtNoise * result.
matSignalCov * invSqrtNoise;
184 SelfAdjointEigenSolver<MatrixXd> signalEig(whitenedSignal);
185 if (signalEig.info() != Success) {
186 qWarning() <<
"Xdawn::fit: signal covariance eigendecomposition failed.";
191 const MatrixXd signalVecsAsc = signalEig.eigenvectors().rightCols(nComponents);
192 MatrixXd signalVecs(signalVecsAsc.rows(), signalVecsAsc.cols());
193 for (
int i = 0; i < nComponents; ++i) {
194 signalVecs.col(i) = signalVecsAsc.col(nComponents - 1 - i);
197 result.
matFilters = invSqrtNoise * signalVecs;
199 for (
int col = 0; col < result.
matFilters.cols(); ++col) {
200 const double noiseNorm = std::sqrt(result.
matFilters.col(col).transpose()
203 if (noiseNorm > 1e-12) {