##// END OF EJS Templates
Remove variable from graph (3)...
Alexandre Leroux -
r271:7858a36671f7
parent child
Show More
@@ -1,226 +1,237
1 1 #include "Visualization/VisualizationGraphWidget.h"
2 2 #include "Visualization/IVisualizationWidgetVisitor.h"
3 3 #include "Visualization/VisualizationGraphHelper.h"
4 4 #include "ui_VisualizationGraphWidget.h"
5 5
6 6 #include <Data/ArrayData.h>
7 7 #include <Data/IDataSeries.h>
8 8 #include <SqpApplication.h>
9 9 #include <Variable/Variable.h>
10 10 #include <Variable/VariableController.h>
11 11
12 12 #include <unordered_map>
13 13
14 14 Q_LOGGING_CATEGORY(LOG_VisualizationGraphWidget, "VisualizationGraphWidget")
15 15
16 16 namespace {
17 17
18 18 /// Key pressed to enable zoom on horizontal axis
19 19 const auto HORIZONTAL_ZOOM_MODIFIER = Qt::NoModifier;
20 20
21 21 /// Key pressed to enable zoom on vertical axis
22 22 const auto VERTICAL_ZOOM_MODIFIER = Qt::ControlModifier;
23 23
24 24 } // namespace
25 25
26 26 struct VisualizationGraphWidget::VisualizationGraphWidgetPrivate {
27 27
28 28 // 1 variable -> n qcpplot
29 29 std::multimap<std::shared_ptr<Variable>, QCPAbstractPlottable *> m_VariableToPlotMultiMap;
30 30 };
31 31
32 32 VisualizationGraphWidget::VisualizationGraphWidget(const QString &name, QWidget *parent)
33 33 : QWidget{parent},
34 34 ui{new Ui::VisualizationGraphWidget},
35 35 impl{spimpl::make_unique_impl<VisualizationGraphWidgetPrivate>()}
36 36 {
37 37 ui->setupUi(this);
38 38
39 39 ui->graphNameLabel->setText(name);
40 40
41 41 // 'Close' options : widget is deleted when closed
42 42 setAttribute(Qt::WA_DeleteOnClose);
43 43 connect(ui->closeButton, &QToolButton::clicked, this, &VisualizationGraphWidget::close);
44 44 ui->closeButton->setIcon(sqpApp->style()->standardIcon(QStyle::SP_TitleBarCloseButton));
45 45
46 46 // Set qcpplot properties :
47 47 // - Drag (on x-axis) and zoom are enabled
48 48 // - Mouse wheel on qcpplot is intercepted to determine the zoom orientation
49 49 ui->widget->setInteractions(QCP::iRangeDrag | QCP::iRangeZoom);
50 50 ui->widget->axisRect()->setRangeDrag(Qt::Horizontal);
51 51 connect(ui->widget, &QCustomPlot::mouseWheel, this, &VisualizationGraphWidget::onMouseWheel);
52 52 connect(ui->widget->xAxis, static_cast<void (QCPAxis::*)(const QCPRange &, const QCPRange &)>(
53 53 &QCPAxis::rangeChanged),
54 54 this, &VisualizationGraphWidget::onRangeChanged);
55 55
56 56 // Activates menu when right clicking on the graph
57 57 ui->widget->setContextMenuPolicy(Qt::CustomContextMenu);
58 58 connect(ui->widget, &QCustomPlot::customContextMenuRequested, this,
59 59 &VisualizationGraphWidget::onGraphMenuRequested);
60 60 }
61 61
62 62
63 63 VisualizationGraphWidget::~VisualizationGraphWidget()
64 64 {
65 65 delete ui;
66 66 }
67 67
68 68 void VisualizationGraphWidget::addVariable(std::shared_ptr<Variable> variable)
69 69 {
70 70 // Uses delegate to create the qcpplot components according to the variable
71 71 auto createdPlottables = VisualizationGraphHelper::create(variable, *ui->widget);
72 72
73 73 for (auto createdPlottable : qAsConst(createdPlottables)) {
74 74 impl->m_VariableToPlotMultiMap.insert({variable, createdPlottable});
75 75 }
76 76
77 77 connect(variable.get(), SIGNAL(dataCacheUpdated()), this, SLOT(onDataCacheVariableUpdated()));
78 78 }
79 79
80 80 void VisualizationGraphWidget::removeVariable(std::shared_ptr<Variable> variable) noexcept
81 81 {
82 // Each component associated to the variable :
83 // - is removed from qcpplot (which deletes it)
84 // - is no longer referenced in the map
85 auto componentsIt = impl->m_VariableToPlotMultiMap.equal_range(variable);
86 for (auto it = componentsIt.first; it != componentsIt.second;) {
87 ui->widget->removePlottable(it->second);
88 it = impl->m_VariableToPlotMultiMap.erase(it);
89 }
90
91 // Updates graph
92 ui->widget->replot();
82 93 }
83 94
84 95 void VisualizationGraphWidget::accept(IVisualizationWidgetVisitor *visitor)
85 96 {
86 97 if (visitor) {
87 98 visitor->visit(this);
88 99 }
89 100 else {
90 101 qCCritical(LOG_VisualizationGraphWidget())
91 102 << tr("Can't visit widget : the visitor is null");
92 103 }
93 104 }
94 105
95 106 bool VisualizationGraphWidget::canDrop(const Variable &variable) const
96 107 {
97 108 /// @todo : for the moment, a graph can always accomodate a variable
98 109 Q_UNUSED(variable);
99 110 return true;
100 111 }
101 112
102 113 QString VisualizationGraphWidget::name() const
103 114 {
104 115 return ui->graphNameLabel->text();
105 116 }
106 117
107 118 void VisualizationGraphWidget::onGraphMenuRequested(const QPoint &pos) noexcept
108 119 {
109 120 QMenu graphMenu{};
110 121
111 122 // Iterates on variables (unique keys)
112 123 for (auto it = impl->m_VariableToPlotMultiMap.cbegin(),
113 124 end = impl->m_VariableToPlotMultiMap.cend();
114 125 it != end; it = impl->m_VariableToPlotMultiMap.upper_bound(it->first)) {
115 126 // 'Remove variable' action
116 127 graphMenu.addAction(tr("Remove variable %1").arg(it->first->name()),
117 128 [ this, var = it->first ]() { removeVariable(var); });
118 129 }
119 130
120 131 if (!graphMenu.isEmpty()) {
121 132 graphMenu.exec(mapToGlobal(pos));
122 133 }
123 134 }
124 135
125 136 void VisualizationGraphWidget::onRangeChanged(const QCPRange &t1, const QCPRange &t2)
126 137 {
127 138
128 139 qCDebug(LOG_VisualizationGraphWidget()) << tr("VisualizationGraphWidget::onRangeChanged");
129 140
130 141 for (auto it = impl->m_VariableToPlotMultiMap.cbegin();
131 142 it != impl->m_VariableToPlotMultiMap.cend(); ++it) {
132 143
133 144 auto variable = it->first;
134 145 qCInfo(LOG_VisualizationGraphWidget())
135 146 << tr("TORM: VisualizationGraphWidget::onRangeChanged")
136 147 << variable->dataSeries()->xAxisData()->size();
137 148 auto dateTime = SqpDateTime{t2.lower, t2.upper};
138 149
139 150 if (!variable->contains(dateTime)) {
140 151
141 152 auto variableDateTimeWithTolerance = dateTime;
142 153 if (variable->intersect(dateTime)) {
143 154 auto variableDateTime = variable->dateTime();
144 155 if (variableDateTime.m_TStart < dateTime.m_TStart) {
145 156
146 157 auto diffEndToKeepDelta = dateTime.m_TEnd - variableDateTime.m_TEnd;
147 158 dateTime.m_TStart = variableDateTime.m_TStart + diffEndToKeepDelta;
148 159 // Tolerance have to be added to the right
149 160 // add 10% tolerance for right (end) side
150 161 auto tolerance = 0.1 * (dateTime.m_TEnd - dateTime.m_TStart);
151 162 variableDateTimeWithTolerance.m_TEnd += tolerance;
152 163 }
153 164 if (variableDateTime.m_TEnd > dateTime.m_TEnd) {
154 165 auto diffStartToKeepDelta = dateTime.m_TStart - dateTime.m_TStart;
155 166 dateTime.m_TEnd = variableDateTime.m_TEnd - diffStartToKeepDelta;
156 167 // Tolerance have to be added to the left
157 168 // add 10% tolerance for left (start) side
158 169 auto tolerance = 0.1 * (dateTime.m_TEnd - dateTime.m_TStart);
159 170 variableDateTimeWithTolerance.m_TStart -= tolerance;
160 171 }
161 172 }
162 173 else {
163 174 // add 10% tolerance for each side
164 175 auto tolerance = 0.1 * (dateTime.m_TEnd - dateTime.m_TStart);
165 176 variableDateTimeWithTolerance.m_TStart -= tolerance;
166 177 variableDateTimeWithTolerance.m_TEnd += tolerance;
167 178 }
168 179 variable->setDateTime(dateTime);
169 180
170 181 // CHangement detected, we need to ask controller to request data loading
171 182 sqpApp->variableController().requestDataLoading(variable,
172 183 variableDateTimeWithTolerance);
173 184 }
174 185 }
175 186 }
176 187
177 188 void VisualizationGraphWidget::onMouseWheel(QWheelEvent *event) noexcept
178 189 {
179 190 auto zoomOrientations = QFlags<Qt::Orientation>{};
180 191
181 192 // Lambda that enables a zoom orientation if the key modifier related to this orientation
182 193 // has
183 194 // been pressed
184 195 auto enableOrientation
185 196 = [&zoomOrientations, event](const auto &orientation, const auto &modifier) {
186 197 auto orientationEnabled = event->modifiers().testFlag(modifier);
187 198 zoomOrientations.setFlag(orientation, orientationEnabled);
188 199 };
189 200 enableOrientation(Qt::Vertical, VERTICAL_ZOOM_MODIFIER);
190 201 enableOrientation(Qt::Horizontal, HORIZONTAL_ZOOM_MODIFIER);
191 202
192 203 ui->widget->axisRect()->setRangeZoom(zoomOrientations);
193 204 }
194 205
195 206 void VisualizationGraphWidget::onDataCacheVariableUpdated()
196 207 {
197 208 // NOTE:
198 209 // We don't want to call the method for each component of a variable unitarily, but for
199 210 // all
200 211 // its components at once (eg its three components in the case of a vector).
201 212
202 213 // The unordered_multimap does not do this easily, so the question is whether to:
203 214 // - use an ordered_multimap and the algos of std to group the values by key
204 215 // - use a map (unique keys) and store as values directly the list of components
205 216
206 217 for (auto it = impl->m_VariableToPlotMultiMap.cbegin();
207 218 it != impl->m_VariableToPlotMultiMap.cend(); ++it) {
208 219 auto variable = it->first;
209 220 VisualizationGraphHelper::updateData(QVector<QCPAbstractPlottable *>{} << it->second,
210 221 variable->dataSeries(), variable->dateTime());
211 222 }
212 223 }
213 224
214 225 void VisualizationGraphWidget::updateDisplay(std::shared_ptr<Variable> variable)
215 226 {
216 227 auto abstractPlotableItPair = impl->m_VariableToPlotMultiMap.equal_range(variable);
217 228
218 229 auto abstractPlotableVect = QVector<QCPAbstractPlottable *>{};
219 230
220 231 for (auto it = abstractPlotableItPair.first; it != abstractPlotableItPair.second; ++it) {
221 232 abstractPlotableVect.push_back(it->second);
222 233 }
223 234
224 235 VisualizationGraphHelper::updateData(abstractPlotableVect, variable->dataSeries(),
225 236 variable->dateTime());
226 237 }
General Comments 0
You need to be logged in to leave comments. Login now