53#include <QCoreApplication>
71Eigen::Vector3f applyTransform(
const Eigen::Vector3f &point,
74 if (trans.
isEmpty())
return point;
75 float r[3] = {point.x(), point.y(), point.z()};
77 return Eigen::Vector3f(r[0], r[1], r[2]);
89 m_loaded = (m_evoked.nave != -1 && m_evoked.data.rows() > 0);
97 if (m_loaded && m_evoked.baseline.first == m_evoked.baseline.second) {
99 float tmin = m_evoked.times.size() > 0 ? m_evoked.times(0) : 0.0f;
101 QPair<float,float> bl(tmin, 0.0f);
102 m_evoked.applyBaselineCorrection(bl);
112 if (!m_loaded || (!m_megMapping && !m_eegMapping))
116 if (m_evoked.info.chs.size() != newEvoked.
info.
chs.size())
120 for (
int i = 0; i < m_evoked.info.chs.size(); ++i) {
121 if (m_evoked.info.chs[i].ch_name != newEvoked.
info.
chs[i].ch_name)
126 if (m_evoked.info.bads != newEvoked.
info.
bads)
130 if (m_evoked.info.projs.size() != newEvoked.
info.
projs.size())
143 const QMap<QString, std::shared_ptr<BrainSurface>> &surfaces)
145 if (surfaces.contains(
"bem_head"))
146 return QStringLiteral(
"bem_head");
149 for (
auto it = surfaces.cbegin(); it != surfaces.cend(); ++it) {
150 if (!it.key().startsWith(
"bem_"))
continue;
153 if (fallback.isEmpty())
162 const QMap<QString, std::shared_ptr<BrainSurface>> &surfaces)
164 return surfaces.contains(
"sens_surface_meg")
165 ? QStringLiteral(
"sens_surface_meg")
173 if (targetTicks <= 0)
return 0.0f;
174 const double range =
static_cast<double>(maxVal - minVal);
175 if (range <= 0.0)
return 0.0f;
177 const double raw = range /
static_cast<double>(targetTicks);
178 const double exponent = std::floor(std::log10(raw));
179 const double base = std::pow(10.0, exponent);
180 const double frac = raw / base;
182 double niceFrac = 1.0;
183 if (frac <= 1.0) niceFrac = 1.0;
184 else if (frac <= 2.0) niceFrac = 2.0;
185 else if (frac <= 5.0) niceFrac = 5.0;
186 else niceFrac = 10.0;
188 return static_cast<float>(niceFrac * base);
194 const QMap<QString, std::shared_ptr<BrainSurface>> &surfaces,
196 bool applySensorTrans)
198 if (!m_loaded || m_evoked.isEmpty())
return false;
203 m_megPositions.clear();
204 m_eegPositions.clear();
205 m_megMapping.reset();
206 m_eegMapping.reset();
209 m_megSurfaceKey = m_megOnHead
213 if (m_megOnHead && m_megSurfaceKey.isEmpty()) {
215 if (!m_megSurfaceKey.isEmpty())
216 qWarning() <<
"SensorFieldMapper: Head surface missing, falling back to helmet.";
220 if (m_megSurfaceKey.isEmpty() && m_eegSurfaceKey.isEmpty()) {
221 qWarning() <<
"SensorFieldMapper: No helmet/head surface for field mapping.";
226 bool hasDevHead =
false;
227 QMatrix4x4 devHeadQt;
228 if (!m_evoked.info.dev_head_t.isEmpty() &&
231 !m_evoked.info.dev_head_t.trans.isIdentity()) {
233 for (
int r = 0; r < 4; ++r)
234 for (
int c = 0; c < 4; ++c)
235 devHeadQt(r, c) = m_evoked.info.dev_head_t.trans(r, c);
238 QMatrix4x4 headToMri;
239 if (applySensorTrans && !headToMriTrans.
isEmpty()) {
240 for (
int r = 0; r < 4; ++r)
241 for (
int c = 0; c < 4; ++c)
242 headToMri(r, c) = headToMriTrans.
trans(r, c);
246 QList<FiffChInfo> megChs, eegChs;
247 QStringList megChNames, eegChNames;
249 auto isBad = [
this](
const QString &name) {
250 return m_evoked.info.bads.contains(name);
253 const int nChs = m_evoked.info.chs.size();
254 m_megPick.resize(nChs);
255 m_eegPick.resize(nChs);
256 int nMeg = 0, nEeg = 0;
258 for (
int k = 0; k < nChs; ++k) {
259 const auto &ch = m_evoked.info.chs[k];
260 if (isBad(ch.ch_name))
continue;
262 QVector3D pos(ch.chpos.r0(0), ch.chpos.r0(1), ch.chpos.r0(2));
265 if (hasDevHead) pos = devHeadQt.map(pos);
266 if (applySensorTrans && !headToMriTrans.
isEmpty()) pos = headToMri.map(pos);
267 m_megPick(nMeg++) = k;
268 m_megPositions.push_back(Eigen::Vector3f(pos.x(), pos.y(), pos.z()));
270 megChNames.append(ch.ch_name);
272 if (applySensorTrans && !headToMriTrans.
isEmpty()) pos = headToMri.map(pos);
273 m_eegPick(nEeg++) = k;
274 m_eegPositions.push_back(Eigen::Vector3f(pos.x(), pos.y(), pos.z()));
276 eegChNames.append(ch.ch_name);
280 m_megPick.conservativeResize(nMeg);
281 m_eegPick.conservativeResize(nEeg);
284 constexpr float kIntrad = 0.06f;
285 constexpr float kMegMiss = 1e-4f;
286 constexpr float kEegMiss = 1e-3f;
300 if (!m_megSurfaceKey.isEmpty() && surfaces.contains(m_megSurfaceKey) && !megChs.isEmpty()) {
306 if (norms.rows() != verts.rows()) {
308 const int nTris = idx.size() / 3;
310 Eigen::MatrixX3i tris(nTris, 3);
311 for (
int t = 0; t < nTris; ++t) {
312 tris(t, 0) =
static_cast<int>(idx[t * 3]);
313 tris(t, 1) =
static_cast<int>(idx[t * 3 + 1]);
314 tris(t, 2) =
static_cast<int>(idx[t * 3 + 2]);
320 if (verts.rows() > 0 && norms.rows() == verts.rows()) {
321 const QString coilPath = QCoreApplication::applicationDirPath()
322 +
"/../resources/general/coilDefinitions/coil_def.dat";
328 if (m_megOnHead && !headMri.
isEmpty()) {
334 }
else if (!devHead.
isEmpty()) {
335 devToTarget = devHead;
338 Eigen::Vector3f origin = fittedOrigin;
339 if (m_megOnHead && !headMri.
isEmpty())
340 origin = applyTransform(origin, headMri);
342 auto coils = templates->create_meg_coils(
345 if (coils && coils->ncoil() > 0) {
347 *coils, verts, norms, origin,
348 m_evoked.info, megChNames,
352 qWarning() <<
"MEG coil definitions not found at" << coilPath;
358 if (!m_eegSurfaceKey.isEmpty() && surfaces.contains(m_eegSurfaceKey) && !eegChs.isEmpty()) {
362 if (verts.rows() > 0) {
363 Eigen::Vector3f origin = fittedOrigin;
364 if (!headMri.
isEmpty()) origin = applyTransform(origin, headMri);
368 eegChs, eegChs.size(), headMri);
370 if (eegCoils && eegCoils->ncoil() > 0) {
372 *eegCoils, verts, origin,
373 m_evoked.info, eegChNames,
388 const Eigen::Vector3f fallback(0.0f, 0.0f, 0.04f);
395 auto gatherPoints = [&](
bool includeEeg) -> Eigen::MatrixXd {
396 QVector<Eigen::Vector3d> pts;
397 for (
const auto &dp : info.
dig) {
402 const double x = dp.r[0], y = dp.r[1], z = dp.r[2];
404 if (z < 0.0 && y > 0.0)
406 pts.append(Eigen::Vector3d(x, y, z));
410 Eigen::MatrixXd mat(pts.size(), 3);
411 for (
int i = 0; i < pts.size(); ++i)
412 mat.row(i) = pts[i].transpose();
416 Eigen::MatrixXd points = gatherPoints(
false);
417 if (points.rows() < 4)
418 points = gatherPoints(
true);
419 if (points.rows() < 4) {
420 qWarning() <<
"SensorFieldMapper::fitSphereOrigin: fewer than 4 dig "
421 "points – falling back to default origin (0, 0, 0.04).";
422 if (radius) *radius = 0.0f;
430 const int n =
static_cast<int>(points.rows());
431 Eigen::MatrixXd A(n, 4);
432 Eigen::VectorXd b(n);
433 for (
int i = 0; i < n; ++i) {
434 A(i, 0) = 2.0 * points(i, 0);
435 A(i, 1) = 2.0 * points(i, 1);
436 A(i, 2) = 2.0 * points(i, 2);
438 b(i) = points(i, 0) * points(i, 0)
439 + points(i, 1) * points(i, 1)
440 + points(i, 2) * points(i, 2);
446 Eigen::Matrix4d AtA = A.transpose() * A;
447 Eigen::Vector4d Atb = A.transpose() * b;
450 x = AtA.fullPivLu().solve(Atb);
452 const float cx =
static_cast<float>(x(0));
453 const float cy =
static_cast<float>(x(1));
454 const float cz =
static_cast<float>(x(2));
455 const float R =
static_cast<float>(
456 std::sqrt(x(0) * x(0) + x(1) * x(1) + x(2) * x(2) + x(3)));
458 if (radius) *radius = R;
460 return Eigen::Vector3f(cx, cy, cz);
470 if (!m_loaded || m_evoked.isEmpty())
473 const int nTimes =
static_cast<int>(m_evoked.data.cols());
478 auto peakGfpTime = [&](
const Eigen::VectorXi &pick) ->
int {
479 if (pick.size() == 0 || nTimes == 0)
return 0;
481 double bestSS = -1.0;
482 for (
int t = 0; t < nTimes; ++t) {
484 for (
int i = 0; i < pick.size(); ++i) {
485 double v = m_evoked.data(pick(i), t);
488 if (ss > bestSS) { bestSS = ss; best = t; }
497 if (m_megMapping && m_megMapping->rows() > 0 && m_megPick.size() > 0) {
498 const int tPeak = peakGfpTime(m_megPick);
499 Eigen::VectorXf meas(m_megPick.size());
500 for (
int i = 0; i < m_megPick.size(); ++i)
501 meas(i) =
static_cast<float>(m_evoked.data(m_megPick(i), tPeak));
503 Eigen::VectorXf mapped = (*m_megMapping) * meas;
504 m_megVmax = mapped.cwiseAbs().maxCoeff();
508 if (m_eegMapping && m_eegMapping->rows() > 0 && m_eegPick.size() > 0) {
509 const int tPeak = peakGfpTime(m_eegPick);
510 Eigen::VectorXf meas(m_eegPick.size());
511 for (
int i = 0; i < m_eegPick.size(); ++i)
512 meas(i) =
static_cast<float>(m_evoked.data(m_eegPick(i), tPeak));
514 Eigen::VectorXf mapped = (*m_eegMapping) * meas;
515 m_eegVmax = mapped.cwiseAbs().maxCoeff();
518 if (m_megVmax <= 0.0f) m_megVmax = 1.0f;
519 if (m_eegVmax <= 0.0f) m_eegVmax = 1.0f;
525 QMap<QString, std::shared_ptr<BrainSurface>> &surfaces,
527 const QVector<SubView> &subViews)
529 if (!m_loaded || m_evoked.isEmpty())
return;
532 auto applyMap = [&](
const QString &key,
533 const QString &contourPrefix,
534 const Eigen::VectorXi &pick,
535 const Eigen::MatrixXf *mat,
539 if (key.isEmpty() || !surfaces.contains(key))
return;
541 auto surface = surfaces[key];
542 if (!visible || !mat || pick.size() == 0) {
544 updateContourSurfaces(surfaces, contourPrefix, *surface,
545 QVector<float>(), 0.0f,
false);
548 if (mat->cols() != pick.size()) {
550 updateContourSurfaces(surfaces, contourPrefix, *surface,
551 QVector<float>(), 0.0f,
false);
556 Eigen::VectorXf meas(pick.size());
557 for (
int i = 0; i < pick.size(); ++i)
558 meas(i) =
static_cast<float>(m_evoked.data(pick(i), m_timePoint));
560 Eigen::VectorXf mapped = (*mat) * meas;
564 const float maxAbs = globalMaxAbs;
567 QVector<uint32_t> colors(mapped.size());
568 for (
int i = 0; i < mapped.size(); ++i) {
569 double norm = (mapped(i) / maxAbs) * 0.5 + 0.5;
570 norm = qBound(0.0, norm, 1.0);
572 QRgb rgb = (m_colormap ==
"MNE")
576 uint32_t r = qRed(rgb);
577 uint32_t g = qGreen(rgb);
578 uint32_t b = qBlue(rgb);
581 surface->applySourceEstimateColors(colors);
585 QVector<float> values(mapped.size());
586 for (
int i = 0; i < mapped.size(); ++i)
587 values[i] = mapped(i);
589 constexpr int nContours = 21;
590 float step = (2.0f * maxAbs) /
static_cast<float>(nContours - 1);
591 updateContourSurfaces(surfaces, contourPrefix, *surface,
592 values, step, showContours);
600 for (
int i = 0; i < subViews.size(); ++i) {
601 anyMegField |= subViews[i].visibility.megFieldMap;
602 anyEegField |= subViews[i].visibility.eegFieldMap;
603 anyMegContours |= subViews[i].visibility.megFieldContours;
604 anyEegContours |= subViews[i].visibility.eegFieldContours;
607 applyMap(m_megSurfaceKey, m_megContourPrefix,
608 m_megPick, m_megMapping.get(),
610 anyMegField, anyMegContours);
612 applyMap(m_eegSurfaceKey, m_eegContourPrefix,
613 m_eegPick, m_eegMapping.get(),
615 anyEegField, anyEegContours);
620void SensorFieldMapper::updateContourSurfaces(
621 QMap<QString, std::shared_ptr<BrainSurface>> &surfaces,
622 const QString &prefix,
624 const QVector<float> &values,
629 auto hideContours = [&]() {
630 for (
const auto &suffix : {QStringLiteral(
"_neg"),
631 QStringLiteral(
"_zero"),
632 QStringLiteral(
"_pos")}) {
633 const QString key = prefix + suffix;
634 if (surfaces.contains(key)) surfaces[key]->setVisible(
false);
638 if (!visible || values.isEmpty() || step <= 0.0f) {
644 float minVal = values[0], maxVal = values[0];
645 for (
int i = 1; i < values.size(); ++i) {
646 minVal = std::min(minVal, values[i]);
647 maxVal = std::max(maxVal, values[i]);
651 QVector<float> negLevels, posLevels;
652 const bool hasZero = (minVal < 0.0f && maxVal > 0.0f);
653 for (
float lv = -step; lv >= minVal; lv -= step) negLevels.append(lv);
654 for (
float lv = step; lv <= maxVal; lv += step) posLevels.append(lv);
658 QVector<Eigen::Vector3f> verts;
659 QVector<Eigen::Vector3f> norms;
660 QVector<Eigen::Vector3i> tris;
663 auto addSegment = [](ContourBuf &buf,
664 const QVector3D &p0,
const QVector3D &p1,
665 const QVector3D &normal,
666 float halfW,
float shift) {
667 QVector3D dir = p1 - p0;
668 const float len = dir.length();
669 if (len < 1e-6f)
return;
672 QVector3D binormal = QVector3D::crossProduct(normal, dir);
673 if (binormal.length() < 1e-6f)
674 binormal = QVector3D::crossProduct(QVector3D(0, 1, 0), dir);
675 if (binormal.length() < 1e-6f)
676 binormal = QVector3D::crossProduct(QVector3D(1, 0, 0), dir);
677 binormal.normalize();
679 const QVector3D off = normal * shift;
681 auto toEig = [](
const QVector3D &v) {
682 return Eigen::Vector3f(v.x(), v.y(), v.z());
687 const QVector3D w = binormal * halfW;
688 const int base = buf.verts.size();
689 Eigen::Vector3f n(normal.x(), normal.y(), normal.z());
691 buf.verts.append(toEig(p0 - w + off));
692 buf.verts.append(toEig(p0 + w + off));
693 buf.verts.append(toEig(p1 - w + off));
694 buf.verts.append(toEig(p1 + w + off));
695 buf.norms.append(n); buf.norms.append(n);
696 buf.norms.append(n); buf.norms.append(n);
697 buf.tris.append(Eigen::Vector3i(base, base + 1, base + 2));
698 buf.tris.append(Eigen::Vector3i(base + 1, base + 3, base + 2));
703 const QVector3D h = normal * halfW;
704 const int base = buf.verts.size();
705 Eigen::Vector3f n(binormal.x(), binormal.y(), binormal.z());
707 buf.verts.append(toEig(p0 - h + off));
708 buf.verts.append(toEig(p0 + h + off));
709 buf.verts.append(toEig(p1 - h + off));
710 buf.verts.append(toEig(p1 + h + off));
711 buf.norms.append(n); buf.norms.append(n);
712 buf.norms.append(n); buf.norms.append(n);
713 buf.tris.append(Eigen::Vector3i(base, base + 1, base + 2));
714 buf.tris.append(Eigen::Vector3i(base + 1, base + 3, base + 2));
719 auto buildContours = [&](
const QVector<float> &levels, ContourBuf &buf) {
723 if (rr.rows() == 0 || nn.rows() == 0 || idx.isEmpty())
return;
725 constexpr float shift = 0.001f;
726 constexpr float halfW = 0.0005f;
728 for (
float level : levels) {
729 for (
int t = 0; t + 2 < idx.size(); t += 3) {
730 const int i0 = idx[t], i1 = idx[t + 1], i2 = idx[t + 2];
731 const float v0 = values[i0], v1 = values[i1], v2 = values[i2];
733 QVector3D p0(rr(i0, 0), rr(i0, 1), rr(i0, 2));
734 QVector3D p1(rr(i1, 0), rr(i1, 1), rr(i1, 2));
735 QVector3D p2(rr(i2, 0), rr(i2, 1), rr(i2, 2));
737 QVector3D n0(nn(i0, 0), nn(i0, 1), nn(i0, 2));
738 QVector3D n1(nn(i1, 0), nn(i1, 1), nn(i1, 2));
739 QVector3D n2(nn(i2, 0), nn(i2, 1), nn(i2, 2));
740 QVector3D triN = (n0 + n1 + n2).normalized();
741 if (triN.length() < 1e-6f)
742 triN = QVector3D::crossProduct(p1 - p0, p2 - p0).normalized();
744 QVector<QVector3D> hits;
745 auto checkEdge = [&](
const QVector3D &a,
const QVector3D &b,
746 float va,
float vb) {
747 if (va == vb)
return;
748 float tval = (level - va) / (vb - va);
749 if (tval >= 0.0f && tval < 1.0f)
750 hits.append(a + (b - a) * tval);
752 checkEdge(p0, p1, v0, v1);
753 checkEdge(p1, p2, v1, v2);
754 checkEdge(p2, p0, v2, v0);
756 if (hits.size() == 2)
757 addSegment(buf, hits[0], hits[1], triN, halfW, shift);
762 ContourBuf negBuf, posBuf, zeroBuf;
763 buildContours(negLevels, negBuf);
764 buildContours(posLevels, posBuf);
766 QVector<float> zeroLevels = {0.0f};
767 buildContours(zeroLevels, zeroBuf);
771 auto updateSurf = [&](
const QString &suffix,
772 const ContourBuf &buf,
775 const QString key = prefix + suffix;
776 if (!show || buf.verts.isEmpty()) {
777 if (surfaces.contains(key)) surfaces[key]->setVisible(
false);
781 Eigen::MatrixX3f rr(buf.verts.size(), 3);
782 Eigen::MatrixX3f nn(buf.norms.size(), 3);
783 Eigen::MatrixX3i tris(buf.tris.size(), 3);
784 for (
int i = 0; i < buf.verts.size(); ++i) {
785 rr.row(i) = buf.verts[i];
786 nn.row(i) = buf.norms[i];
788 for (
int i = 0; i < buf.tris.size(); ++i)
789 tris.row(i) = buf.tris[i];
791 std::shared_ptr<BrainSurface> csurf;
792 if (surfaces.contains(key)) {
793 csurf = surfaces[key];
795 csurf = std::make_shared<BrainSurface>();
796 surfaces[key] = csurf;
798 csurf->createFromData(rr, nn, tris, color);
799 csurf->setVisible(
true);
802 updateSurf(
"_neg", negBuf, QColor(0, 0, 255, 200), visible && !negBuf.verts.isEmpty());
803 updateSurf(
"_zero", zeroBuf, QColor(0, 0, 0, 220), visible && !zeroBuf.verts.isEmpty());
804 updateSurf(
"_pos", posBuf, QColor(255, 0, 0, 200), visible && !posBuf.verts.isEmpty());
ColorMap class declaration.
#define FIFFV_POINT_EXTRA
#define FIFFV_COORD_DEVICE
FiffChInfo class declaration.
FiffCoordTrans class declaration.
FsSurface class declaration.
SensorFieldMapper class declaration.
BrainSurface class declaration.
QRgb mneAnalyzeColor(double v)
Lightweight render-related enums shared across the disp3D library.
uint32_t packABGR(uint32_t r, uint32_t g, uint32_t b, uint32_t a=0xFF)
FwdFieldMap class declaration.
FwdCoilSet class declaration.
FIFF file I/O and data structures (raw, epochs, evoked, covariance, forward).
constexpr int FWD_COIL_ACCURACY_NORMAL
static QRgb valueToColor(double v, const QString &sMap)
Viewport subdivision holding its own camera, projection, and scissor rectangle.
ViewVisibilityProfile visibility
Renderable cortical surface mesh with per-vertex color, curvature data, and GPU buffer management.
Eigen::MatrixX3f vertexNormals() const
QVector< uint32_t > triangleIndices() const
Eigen::MatrixX3f vertexPositions() const
static constexpr VisualizationMode ModeSurface
static QString findHeadSurfaceKey(const QMap< QString, std::shared_ptr< BrainSurface > > &surfaces)
static float contourStep(float minVal, float maxVal, int targetTicks)
bool hasMappingFor(const FIFFLIB::FiffEvoked &newEvoked) const
const FIFFLIB::FiffEvoked & evoked() const
static Eigen::Vector3f fitSphereOrigin(const FIFFLIB::FiffInfo &info, float *radius=nullptr)
bool buildMapping(const QMap< QString, std::shared_ptr< BrainSurface > > &surfaces, const FIFFLIB::FiffCoordTrans &headToMriTrans, bool applySensorTrans)
static QString findHelmetSurfaceKey(const QMap< QString, std::shared_ptr< BrainSurface > > &surfaces)
void setEvoked(const FIFFLIB::FiffEvoked &evoked)
void apply(QMap< QString, std::shared_ptr< BrainSurface > > &surfaces, const SubView &singleView, const QVector< SubView > &subViews)
Coordinate transformation description.
Eigen::Matrix< float, 4, 4, Eigen::DontAlign > trans
FIFF measurement file information.
QList< FiffDigPoint > dig
FiffCoordTrans dev_head_t
static Eigen::MatrixX3f compute_normals(const Eigen::MatrixX3f &rr, const Eigen::MatrixX3i &tris)
static FwdCoilSet::UPtr read_coil_defs(const QString &name)
static FwdCoilSet::UPtr create_eeg_els(const QList< FIFFLIB::FiffChInfo > &chs, int nch, const FIFFLIB::FiffCoordTrans &t=FIFFLIB::FiffCoordTrans())
static std::unique_ptr< Eigen::MatrixXf > computeEegMapping(const FwdCoilSet &coils, const Eigen::MatrixX3f &vertices, const Eigen::Vector3f &origin, float intrad=0.06f, float miss=1e-3f)
static std::unique_ptr< Eigen::MatrixXf > computeMegMapping(const FwdCoilSet &coils, const Eigen::MatrixX3f &vertices, const Eigen::MatrixX3f &normals, const Eigen::Vector3f &origin, float intrad=0.06f, float miss=1e-4f)
static FiffCoordTrans combine(int from, int to, const FiffCoordTrans &t1, const FiffCoordTrans &t2)
Eigen::MatrixX3f apply_trans(const Eigen::MatrixX3f &rr, bool do_move=true) const