/* * Copyright (C) 2016, Mike Walters * * This file is part of inspectrum. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include "samplesource.h" #include "traceplot.h" TracePlot::TracePlot(std::shared_ptr source) : Plot(source) { connect(this, &TracePlot::imageReady, this, &TracePlot::handleImage); } void TracePlot::paintMid(QPainter &painter, QRect &rect, range_t sampleRange) { if (sampleRange.length() == 0) return; int samplesPerColumn = sampleRange.length() / rect.width(); int samplesPerTile = tileWidth * samplesPerColumn; size_t tileID = sampleRange.minimum / samplesPerTile; size_t tileOffset = sampleRange.minimum % samplesPerTile; // Number of samples to skip from first image tile int xOffset = tileOffset / samplesPerColumn; // Number of columns to skip from first image tile // Paint first (possibly partial) tile painter.drawPixmap( QRect(rect.x(), rect.y(), tileWidth - xOffset, height()), getTile(tileID++, samplesPerTile), QRect(xOffset, 0, tileWidth - xOffset, height()) ); // Paint remaining tiles for (int x = tileWidth - xOffset; x < rect.right(); x += tileWidth) { painter.drawPixmap( QRect(x, rect.y(), tileWidth, height()), getTile(tileID++, samplesPerTile) ); } } QPixmap TracePlot::getTile(size_t tileID, size_t sampleCount) { QPixmap pixmap(tileWidth, height()); QString key; QTextStream(&key) << "traceplot_" << this << "_" << tileID << "_" << sampleCount; if (QPixmapCache::find(key, &pixmap)) return pixmap; if (!tasks.contains(key)) { range_t sampleRange{tileID * sampleCount, (tileID + 1) * sampleCount}; QtConcurrent::run(this, &TracePlot::drawTile, key, QRect(0, 0, tileWidth, height()), sampleRange); tasks.insert(key); } pixmap.fill(Qt::transparent); return pixmap; } void TracePlot::drawTile(QString key, const QRect &rect, range_t sampleRange) { QImage image(rect.size(), QImage::Format_ARGB32); image.fill(Qt::transparent); QPainter painter(&image); painter.setRenderHint(QPainter::Antialiasing, true); auto firstSample = sampleRange.minimum; auto length = sampleRange.length(); // Is it a 2-channel (complex) trace? if (auto src = dynamic_cast>*>(sampleSource.get())) { auto samples = src->getSamples(firstSample, length); if (samples == nullptr) return; painter.setPen(Qt::red); plotTrace(painter, rect, reinterpret_cast(samples.get()), length, 2); painter.setPen(Qt::blue); plotTrace(painter, rect, reinterpret_cast(samples.get())+1, length, 2); // Otherwise is it single channel? } else if (auto src = dynamic_cast*>(sampleSource.get())) { auto samples = src->getSamples(firstSample, length); if (samples == nullptr) return; painter.setPen(Qt::green); plotTrace(painter, rect, samples.get(), length, 1); } else { throw std::runtime_error("TracePlot::paintMid: Unsupported source type"); } emit imageReady(key, image); } void TracePlot::handleImage(QString key, QImage image) { auto pixmap = QPixmap::fromImage(image); QPixmapCache::insert(key, pixmap); tasks.remove(key); emit repaint(); } void TracePlot::plotTrace(QPainter &painter, const QRect &rect, float *samples, size_t count, int step = 1) { QPainterPath path; range_t xRange{0, rect.width() - 2.f}; range_t yRange{0, rect.height() - 2.f}; const float xStep = 1.0 / count * rect.width(); for (size_t i = 0; i < count; i++) { float sample = samples[i*step]; float x = i * xStep; float y = (1 - sample) * (rect.height() / 2); x = xRange.clip(x) + rect.x(); y = yRange.clip(y) + rect.y(); if (i == 0) path.moveTo(x, y); else path.lineTo(x, y); } painter.drawPath(path); }