VisualizationGraphHelper.cpp
247 lines
| 8.3 KiB
| text/x-c
|
CppLexer
r227 | #include "Visualization/VisualizationGraphHelper.h" | |||
Alexandre Leroux
|
r168 | #include "Visualization/qcustomplot.h" | ||
Alexandre Leroux
|
r548 | #include <Common/ColorUtils.h> | ||
Alexandre Leroux
|
r169 | #include <Data/ScalarSeries.h> | ||
Alexandre Leroux
|
r547 | #include <Data/VectorSeries.h> | ||
Alexandre Leroux
|
r169 | |||
Alexandre Leroux
|
r168 | #include <Variable/Variable.h> | ||
r227 | Q_LOGGING_CATEGORY(LOG_VisualizationGraphHelper, "VisualizationGraphHelper") | |||
Alexandre Leroux
|
r168 | |||
Alexandre Leroux
|
r169 | namespace { | ||
r336 | class SqpDataContainer : public QCPGraphDataContainer { | |||
public: | ||||
Alexandre Leroux
|
r419 | void appendGraphData(const QCPGraphData &data) { mData.append(data); } | ||
r336 | }; | |||
Alexandre Leroux
|
r170 | /// Format for datetimes on a axis | ||
const auto DATETIME_TICKER_FORMAT = QStringLiteral("yyyy/MM/dd \nhh:mm:ss"); | ||||
/// Generates the appropriate ticker for an axis, depending on whether the axis displays time or | ||||
/// non-time data | ||||
QSharedPointer<QCPAxisTicker> axisTicker(bool isTimeAxis) | ||||
{ | ||||
if (isTimeAxis) { | ||||
auto dateTicker = QSharedPointer<QCPAxisTickerDateTime>::create(); | ||||
dateTicker->setDateTimeFormat(DATETIME_TICKER_FORMAT); | ||||
Alexandre Leroux
|
r452 | dateTicker->setDateTimeSpec(Qt::UTC); | ||
Alexandre Leroux
|
r170 | |||
return dateTicker; | ||||
} | ||||
else { | ||||
// default ticker | ||||
return QSharedPointer<QCPAxisTicker>::create(); | ||||
} | ||||
} | ||||
Alexandre Leroux
|
r169 | |||
r594 | /// Sets axes properties according to the properties of a data series. Not thread safe | |||
Alexandre Leroux
|
r547 | template <int Dim> | ||
void setAxesProperties(const DataSeries<Dim> &dataSeries, QCustomPlot &plot) noexcept | ||||
r219 | { | |||
Alexandre Leroux
|
r547 | /// @todo : for the moment, no control is performed on the axes: the units and the tickers | ||
/// are fixed for the default x-axis and y-axis of the plot, and according to the new graph | ||||
auto setAxisProperties = [](auto axis, const auto &unit) { | ||||
// label (unit name) | ||||
axis->setLabel(unit.m_Name); | ||||
// ticker (depending on the type of unit) | ||||
axis->setTicker(axisTicker(unit.m_TimeUnit)); | ||||
}; | ||||
setAxisProperties(plot.xAxis, dataSeries.xAxisUnit()); | ||||
setAxisProperties(plot.yAxis, dataSeries.valuesUnit()); | ||||
} | ||||
r290 | ||||
Alexandre Leroux
|
r547 | /** | ||
r554 | * Struct used to create plottables, depending on the type of the data series from which to create | |||
* them | ||||
Alexandre Leroux
|
r547 | * @tparam T the data series' type | ||
* @remarks Default implementation can't create plottables | ||||
*/ | ||||
template <typename T, typename Enabled = void> | ||||
struct PlottablesCreator { | ||||
static PlottablesMap createPlottables(T &, QCustomPlot &) | ||||
{ | ||||
qCCritical(LOG_DataSeries()) | ||||
<< QObject::tr("Can't create plottables: unmanaged data series type"); | ||||
return {}; | ||||
r219 | } | |||
Alexandre Leroux
|
r547 | }; | ||
r219 | ||||
Alexandre Leroux
|
r547 | /** | ||
* Specialization of PlottablesCreator for scalars and vectors | ||||
* @sa ScalarSeries | ||||
* @sa VectorSeries | ||||
*/ | ||||
template <typename T> | ||||
struct PlottablesCreator<T, | ||||
typename std::enable_if_t<std::is_base_of<ScalarSeries, T>::value | ||||
or std::is_base_of<VectorSeries, T>::value> > { | ||||
static PlottablesMap createPlottables(T &dataSeries, QCustomPlot &plot) | ||||
{ | ||||
PlottablesMap result{}; | ||||
// Gets the number of components of the data series | ||||
r594 | dataSeries.lockRead(); | |||
Alexandre Leroux
|
r547 | auto componentCount = dataSeries.valuesData()->componentCount(); | ||
r594 | dataSeries.unlock(); | |||
Alexandre Leroux
|
r547 | |||
Alexandre Leroux
|
r548 | auto colors = ColorUtils::colors(Qt::blue, Qt::red, componentCount); | ||
Alexandre Leroux
|
r547 | // For each component of the data series, creates a QCPGraph to add to the plot | ||
for (auto i = 0; i < componentCount; ++i) { | ||||
auto graph = plot.addGraph(); | ||||
Alexandre Leroux
|
r548 | graph->setPen(QPen{colors.at(i)}); | ||
Alexandre Leroux
|
r547 | |||
result.insert({i, graph}); | ||||
} | ||||
r518 | ||||
// Axes properties | ||||
r594 | dataSeries.lockRead(); | |||
Alexandre Leroux
|
r547 | setAxesProperties(dataSeries, plot); | ||
r594 | dataSeries.unlock(); | |||
Alexandre Leroux
|
r547 | |||
plot.replot(); | ||||
return result; | ||||
r518 | } | |||
Alexandre Leroux
|
r547 | }; | ||
r518 | ||||
Alexandre Leroux
|
r547 | /** | ||
r554 | * Struct used to update plottables, depending on the type of the data series from which to update | |||
* them | ||||
Alexandre Leroux
|
r547 | * @tparam T the data series' type | ||
* @remarks Default implementation can't update plottables | ||||
*/ | ||||
template <typename T, typename Enabled = void> | ||||
struct PlottablesUpdater { | ||||
static void updatePlottables(T &, PlottablesMap &, const SqpRange &, bool) | ||||
{ | ||||
qCCritical(LOG_DataSeries()) | ||||
<< QObject::tr("Can't update plottables: unmanaged data series type"); | ||||
} | ||||
}; | ||||
Alexandre Leroux
|
r169 | |||
Alexandre Leroux
|
r547 | /** | ||
* Specialization of PlottablesUpdater for scalars and vectors | ||||
* @sa ScalarSeries | ||||
* @sa VectorSeries | ||||
*/ | ||||
template <typename T> | ||||
struct PlottablesUpdater<T, | ||||
typename std::enable_if_t<std::is_base_of<ScalarSeries, T>::value | ||||
or std::is_base_of<VectorSeries, T>::value> > { | ||||
static void updatePlottables(T &dataSeries, PlottablesMap &plottables, const SqpRange &range, | ||||
bool rescaleAxes) | ||||
{ | ||||
// For each plottable to update, resets its data | ||||
std::map<int, QSharedPointer<SqpDataContainer> > dataContainers{}; | ||||
for (const auto &plottable : plottables) { | ||||
if (auto graph = dynamic_cast<QCPGraph *>(plottable.second)) { | ||||
auto dataContainer = QSharedPointer<SqpDataContainer>::create(); | ||||
graph->setData(dataContainer); | ||||
dataContainers.insert({plottable.first, dataContainer}); | ||||
} | ||||
} | ||||
r594 | dataSeries.lockRead(); | |||
Alexandre Leroux
|
r169 | |||
Alexandre Leroux
|
r547 | // - Gets the data of the series included in the current range | ||
r554 | // - Updates each plottable by adding, for each data item, a point that takes x-axis data | |||
// and value data. The correct value is retrieved according to the index of the component | ||||
Alexandre Leroux
|
r566 | auto subDataIts = dataSeries.xAxisRange(range.m_TStart, range.m_TEnd); | ||
Alexandre Leroux
|
r547 | for (auto it = subDataIts.first; it != subDataIts.second; ++it) { | ||
for (const auto &dataContainer : dataContainers) { | ||||
auto componentIndex = dataContainer.first; | ||||
dataContainer.second->appendGraphData( | ||||
QCPGraphData(it->x(), it->value(componentIndex))); | ||||
} | ||||
} | ||||
r219 | ||||
Alexandre Leroux
|
r547 | dataSeries.unlock(); | ||
Alexandre Leroux
|
r170 | |||
Alexandre Leroux
|
r547 | if (!plottables.empty()) { | ||
auto plot = plottables.begin()->second->parentPlot(); | ||||
Alexandre Leroux
|
r170 | |||
Alexandre Leroux
|
r547 | if (rescaleAxes) { | ||
plot->rescaleAxes(); | ||||
} | ||||
Alexandre Leroux
|
r170 | |||
Alexandre Leroux
|
r547 | plot->replot(); | ||
} | ||||
Alexandre Leroux
|
r169 | } | ||
Alexandre Leroux
|
r547 | }; | ||
/** | ||||
* Helper used to create/update plottables | ||||
*/ | ||||
struct IPlottablesHelper { | ||||
virtual ~IPlottablesHelper() noexcept = default; | ||||
virtual PlottablesMap create(QCustomPlot &plot) const = 0; | ||||
virtual void update(PlottablesMap &plottables, const SqpRange &range, | ||||
bool rescaleAxes = false) const = 0; | ||||
}; | ||||
/** | ||||
* Default implementation of IPlottablesHelper, which takes data series to create/update plottables | ||||
* @tparam T the data series' type | ||||
*/ | ||||
template <typename T> | ||||
struct PlottablesHelper : public IPlottablesHelper { | ||||
explicit PlottablesHelper(T &dataSeries) : m_DataSeries{dataSeries} {} | ||||
PlottablesMap create(QCustomPlot &plot) const override | ||||
{ | ||||
return PlottablesCreator<T>::createPlottables(m_DataSeries, plot); | ||||
Alexandre Leroux
|
r169 | } | ||
Alexandre Leroux
|
r547 | void update(PlottablesMap &plottables, const SqpRange &range, bool rescaleAxes) const override | ||
{ | ||||
PlottablesUpdater<T>::updatePlottables(m_DataSeries, plottables, range, rescaleAxes); | ||||
} | ||||
Alexandre Leroux
|
r169 | |||
Alexandre Leroux
|
r547 | T &m_DataSeries; | ||
}; | ||||
Alexandre Leroux
|
r169 | |||
Alexandre Leroux
|
r547 | /// Creates IPlottablesHelper according to a data series | ||
Alexandre Leroux
|
r551 | std::unique_ptr<IPlottablesHelper> createHelper(std::shared_ptr<IDataSeries> dataSeries) noexcept | ||
r518 | { | |||
Alexandre Leroux
|
r551 | if (auto scalarSeries = std::dynamic_pointer_cast<ScalarSeries>(dataSeries)) { | ||
Alexandre Leroux
|
r547 | return std::make_unique<PlottablesHelper<ScalarSeries> >(*scalarSeries); | ||
} | ||||
Alexandre Leroux
|
r551 | else if (auto vectorSeries = std::dynamic_pointer_cast<VectorSeries>(dataSeries)) { | ||
Alexandre Leroux
|
r547 | return std::make_unique<PlottablesHelper<VectorSeries> >(*vectorSeries); | ||
r518 | } | |||
else { | ||||
Alexandre Leroux
|
r547 | return std::make_unique<PlottablesHelper<IDataSeries> >(*dataSeries); | ||
r518 | } | |||
} | ||||
Alexandre Leroux
|
r547 | } // namespace | ||
Alexandre Leroux
|
r168 | |||
Alexandre Leroux
|
r547 | PlottablesMap VisualizationGraphHelper::create(std::shared_ptr<Variable> variable, | ||
QCustomPlot &plot) noexcept | ||||
{ | ||||
Alexandre Leroux
|
r168 | if (variable) { | ||
Alexandre Leroux
|
r551 | auto helper = createHelper(variable->dataSeries()); | ||
Alexandre Leroux
|
r547 | auto plottables = helper->create(plot); | ||
return plottables; | ||||
Alexandre Leroux
|
r168 | } | ||
else { | ||||
r227 | qCDebug(LOG_VisualizationGraphHelper()) | |||
Alexandre Leroux
|
r168 | << QObject::tr("Can't create graph plottables : the variable is null"); | ||
Alexandre Leroux
|
r547 | return PlottablesMap{}; | ||
Alexandre Leroux
|
r168 | } | ||
} | ||||
r219 | ||||
Alexandre Leroux
|
r551 | void VisualizationGraphHelper::updateData(PlottablesMap &plottables, | ||
std::shared_ptr<IDataSeries> dataSeries, | ||||
r513 | const SqpRange &dateTime) | |||
r219 | { | |||
Alexandre Leroux
|
r547 | auto helper = createHelper(dataSeries); | ||
helper->update(plottables, dateTime); | ||||
r219 | } | |||