65 setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
66 setCursor(Qt::PointingHandCursor);
88 m_envelopeDirty =
true;
96 m_firstFileSample = first;
97 m_envelopeDirty =
true;
105 m_lastFileSample = last;
106 m_envelopeDirty =
true;
122 m_scrollSample = scrollSample;
123 m_visibleSamples = visibleSamples;
139 m_annotations = annotations;
145float OverviewBarWidget::xToSample(
int x)
const
147 int totalSamples = m_lastFileSample - m_firstFileSample;
148 if (totalSamples <= 0 || width() <= 0)
149 return static_cast<float>(m_firstFileSample);
150 float frac =
static_cast<float>(x) /
static_cast<float>(width());
151 frac = qBound(0.f, frac, 1.f);
152 return static_cast<float>(m_firstFileSample) + frac *
static_cast<float>(totalSamples);
157void OverviewBarWidget::rebuildEnvelope()
159 const int pw = width();
161 if (pw <= 0 || !m_model || m_lastFileSample <= m_firstFileSample) {
162 m_envelopeImage = QImage();
163 m_envelopeDirty =
false;
167 m_envelopeImage = QImage(pw, ph, QImage::Format_RGB32);
168 m_envelopeImage.fill(QColor(38, 38, 42).rgb());
170 QPainter p(&m_envelopeImage);
171 p.setRenderHint(QPainter::Antialiasing,
false);
173 const int totalSamples = m_lastFileSample - m_firstFileSample;
174 const float samplesPerPx =
static_cast<float>(totalSamples) /
static_cast<float>(pw);
175 const int nChannels = m_model->channelCount();
176 if (nChannels <= 0 || totalSamples <= 0) {
177 m_envelopeDirty =
false;
182 struct TypeEnvelope {
185 QVector<float> envelope;
187 QVector<TypeEnvelope> typeEnvelopes;
188 QMap<QString, int> typeIndex;
190 for (
int ch = 0; ch < nChannels; ++ch) {
191 auto info = m_model->channelInfo(ch);
192 if (!typeIndex.contains(info.typeLabel)) {
193 int idx = typeEnvelopes.size();
194 typeIndex[info.typeLabel] = idx;
196 te.typeLabel = info.typeLabel;
197 te.color = info.color;
198 te.envelope.resize(pw, 0.f);
199 typeEnvelopes.append(te);
204 for (
int px_col = 0; px_col < pw; ++px_col) {
205 int sampleStart = m_firstFileSample +
static_cast<int>(px_col * samplesPerPx);
206 int sampleEnd = m_firstFileSample +
static_cast<int>((px_col + 1) * samplesPerPx);
207 sampleEnd = qMin(sampleEnd, m_lastFileSample);
208 if (sampleEnd <= sampleStart)
212 int step = qMax(1, (sampleEnd - sampleStart) / 4);
213 for (
int ch = 0; ch < nChannels; ++ch) {
214 auto info = m_model->channelInfo(ch);
215 int ti = typeIndex.value(info.typeLabel, -1);
216 if (ti < 0)
continue;
219 for (
int s = sampleStart; s < sampleEnd; s += step) {
220 float v = qAbs(m_model->sampleValueAt(ch, s));
221 if (v > maxAbs) maxAbs = v;
224 float norm = (info.amplitudeMax > 0.f) ? maxAbs / info.amplitudeMax : 0.f;
225 if (norm > typeEnvelopes[ti].envelope[px_col])
226 typeEnvelopes[ti].envelope[px_col] = norm;
231 const int nTypes = typeEnvelopes.size();
232 const float laneH =
static_cast<float>(ph) / qMax(nTypes, 1);
234 for (
int ti = 0; ti < nTypes; ++ti) {
235 const auto &te = typeEnvelopes[ti];
236 QColor fillColor = te.color;
237 fillColor.setAlpha(160);
240 float yBase = (ti + 1) * laneH;
241 for (
int x = 0; x < pw; ++x) {
242 float level = qBound(0.f, te.envelope[x], 1.f);
243 float barH = level * laneH * 0.9f;
244 if (barH < 0.5f)
continue;
245 p.fillRect(QRectF(x, yBase - barH, 1.f, barH), fillColor);
250 f.setPointSizeF(7.0);
252 p.setPen(QColor(200, 200, 200, 180));
253 p.drawText(QRectF(2, ti * laneH, 60, laneH * 0.5f),
254 Qt::AlignLeft | Qt::AlignTop, te.typeLabel);
257 m_envelopeDirty =
false;
265 p.setRenderHint(QPainter::Antialiasing,
false);
267 const int pw = width();
268 const int ph = height();
270 if (m_envelopeDirty || m_envelopeImage.width() != pw)
274 if (!m_envelopeImage.isNull()) {
275 p.drawImage(0, 0, m_envelopeImage);
277 p.fillRect(rect(), QColor(38, 38, 42));
280 int totalSamples = m_lastFileSample - m_firstFileSample;
281 if (totalSamples <= 0) {
283 p.setPen(QColor(120, 120, 130));
284 p.drawText(rect(), Qt::AlignCenter, QStringLiteral(
"No data"));
288 float samplesPerPx =
static_cast<float>(totalSamples) /
static_cast<float>(pw);
291 for (
const auto &ann : m_annotations) {
292 float xStart =
static_cast<float>(ann.startSample - m_firstFileSample) / samplesPerPx;
293 float xEnd =
static_cast<float>(ann.endSample - m_firstFileSample) / samplesPerPx;
294 xStart = qBound(0.f, xStart,
static_cast<float>(pw));
295 xEnd = qBound(0.f, xEnd,
static_cast<float>(pw));
297 QColor c = ann.color;
299 p.fillRect(QRectF(xStart, 0, xEnd - xStart, ph), c);
304 for (
const auto &ev : m_events) {
305 float xF =
static_cast<float>(ev.sample - m_firstFileSample) / samplesPerPx;
306 if (xF < 0.f || xF > pw)
310 p.setPen(QPen(c, 1));
311 int ix =
static_cast<int>(xF);
312 p.drawLine(ix, ph - 6, ix, ph);
316 float vpX = (m_scrollSample -
static_cast<float>(m_firstFileSample)) / samplesPerPx;
317 float vpW = m_visibleSamples / samplesPerPx;
318 vpX = qBound(0.f, vpX,
static_cast<float>(pw));
319 vpW = qBound(2.f, vpW,
static_cast<float>(pw) - vpX);
322 QColor dimColor(0, 0, 0, 100);
324 p.fillRect(QRectF(0, 0, vpX, ph), dimColor);
326 p.fillRect(QRectF(vpX + vpW, 0, pw - vpX - vpW, ph), dimColor);
329 QPen vpPen(QColor(255, 255, 255, 200), 1.5);
331 p.setBrush(Qt::NoBrush);
332 p.drawRect(QRectF(vpX, 0.5f, vpW, ph - 1.f));
339 if (event->button() == Qt::LeftButton) {
341 float targetSample = xToSample(event->position().toPoint().x()) - m_visibleSamples * 0.5f;
352 float targetSample = xToSample(event->position().toPoint().x()) - m_visibleSamples * 0.5f;
362 if (event->button() == Qt::LeftButton) {
Declaration of the OverviewBarWidget class.
Declaration of the ChannelDataModel class.
2-D display widgets and visualisation helpers (charts, topography, colour maps).
ChannelDataModel – lightweight data container for ChannelDataView / ChannelRhiView.
void setFirstFileSample(int first)
void setLastFileSample(int last)
void scrollRequested(float targetSample)
void paintEvent(QPaintEvent *event) override
QSize sizeHint() const override
QSize minimumSizeHint() const override
void setAnnotations(const QVector< ChannelRhiView::AnnotationSpan > &annotations)
void setModel(ChannelDataModel *model)
void setSfreq(float sfreq)
void mouseMoveEvent(QMouseEvent *event) override
void setEvents(const QVector< ChannelRhiView::EventMarker > &events)
void mousePressEvent(QMouseEvent *event) override
void mouseReleaseEvent(QMouseEvent *event) override
OverviewBarWidget(QWidget *parent=nullptr)
void setViewport(float scrollSample, float visibleSamples)
static constexpr int kBarHeight