PulseView  unreleased development snapshot
A Qt-based sigrok GUI
cursorpair.cpp
Go to the documentation of this file.
1 /*
2  * This file is part of the PulseView project.
3  *
4  * Copyright (C) 2013 Joel Holdsworth <joel@airwebreathe.org.uk>
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, see <http://www.gnu.org/licenses/>.
18  */
19 
20 #include <algorithm>
21 #include <cassert>
22 
23 #include <QColor>
24 #include <QMenu>
25 #include <QToolTip>
26 
27 #include "cursorpair.hpp"
28 
29 #include "pv/globalsettings.hpp"
30 #include "pv/util.hpp"
31 #include "ruler.hpp"
32 #include "view.hpp"
33 
34 using std::max;
35 using std::make_pair;
36 using std::min;
37 using std::shared_ptr;
38 using std::pair;
39 
40 namespace pv {
41 namespace views {
42 namespace trace {
43 
44 const int CursorPair::DeltaPadding = 8;
45 
47  TimeItem(view),
48  first_(new Cursor(view, 0.0)),
49  second_(new Cursor(view, 1.0)),
50  label_incomplete_(true)
51 {
53 
54  GlobalSettings settings;
55  fill_color_ = QColor::fromRgba(settings.value(
56  GlobalSettings::Key_View_CursorFillColor).value<uint32_t>());
57  show_frequency_ = settings.value(
59  show_interval_ = settings.value(
61  show_samples_ = settings.value(
63 
64  connect(&view_, SIGNAL(hover_point_changed(const QWidget*, QPoint)),
65  this, SLOT(on_hover_point_changed(const QWidget*, QPoint)));
66 }
67 
69 {
71 }
72 
73 bool CursorPair::enabled() const
74 {
75  return view_.cursors_shown();
76 }
77 
78 shared_ptr<Cursor> CursorPair::first() const
79 {
80  return first_;
81 }
82 
83 shared_ptr<Cursor> CursorPair::second() const
84 {
85  return second_;
86 }
87 
89 {
90  const pv::util::Timestamp delta = second_->time() - first_->time();
91  first_->set_time(time);
92  second_->set_time(time + delta);
93 }
94 
96 {
97  return 0;
98 }
99 
100 float CursorPair::get_x() const
101 {
102  return (first_->get_x() + second_->get_x()) / 2.0f;
103 }
104 
106 {
107  if (other < second_->time())
108  return other - first_->time();
109  else
110  return other - second_->time();
111 }
112 
113 QPoint CursorPair::drag_point(const QRect &rect) const
114 {
115  return first_->drag_point(rect);
116 }
117 
119 {
120  (void)parent;
121  return nullptr;
122 }
123 
125 {
126  QMenu *menu = new QMenu(parent);
127 
128  QAction *displayIntervalAction = new QAction(tr("Display interval"), this);
129  displayIntervalAction->setCheckable(true);
130  displayIntervalAction->setChecked(show_interval_);
131  menu->addAction(displayIntervalAction);
132 
133  connect(displayIntervalAction, &QAction::toggled, displayIntervalAction,
134  [=]{
135  GlobalSettings settings;
137  !settings.value(GlobalSettings::Key_View_CursorShowInterval).value<bool>());
138  });
139 
140  QAction *displayFrequencyAction = new QAction(tr("Display frequency"), this);
141  displayFrequencyAction->setCheckable(true);
142  displayFrequencyAction->setChecked(show_frequency_);
143  menu->addAction(displayFrequencyAction);
144 
145  connect(displayFrequencyAction, &QAction::toggled, displayFrequencyAction,
146  [=]{
147  GlobalSettings settings;
149  !settings.value(GlobalSettings::Key_View_CursorShowFrequency).value<bool>());
150  });
151 
152  QAction *displaySamplesAction = new QAction(tr("Display samples"), this);
153  displaySamplesAction->setCheckable(true);
154  displaySamplesAction->setChecked(show_samples_);
155  menu->addAction(displaySamplesAction);
156 
157  connect(displaySamplesAction, &QAction::toggled, displaySamplesAction,
158  [=]{
159  GlobalSettings settings;
161  !settings.value(GlobalSettings::Key_View_CursorShowSamples).value<bool>());
162  });
163 
164  return menu;
165 }
166 
167 QRectF CursorPair::label_rect(const QRectF &rect) const
168 {
169  const QSizeF label_size(text_size_ + LabelPadding * 2);
170  const pair<float, float> offsets(get_cursor_offsets());
171  const pair<float, float> normal_offsets(
172  (offsets.first < offsets.second) ? offsets :
173  make_pair(offsets.second, offsets.first));
174 
175  const float height = label_size.height();
176  const float left = max(normal_offsets.first + DeltaPadding, -height);
177  const float right = min(normal_offsets.second - DeltaPadding,
178  (float)rect.width() + height);
179 
180  return QRectF(left, rect.height() - label_size.height() -
181  TimeMarker::ArrowSize - 0.5f,
182  right - left, height);
183 }
184 
185 void CursorPair::paint_label(QPainter &p, const QRect &rect, bool hover)
186 {
187  assert(first_);
188  assert(second_);
189 
190  if (!enabled())
191  return;
192 
193  const QColor text_color = ViewItem::select_text_color(Cursor::FillColor);
194  p.setPen(text_color);
195 
196  QRectF delta_rect(label_rect(rect));
197  const int radius = delta_rect.height() / 2;
198  QRectF text_rect(delta_rect.intersected(rect).adjusted(radius, 0, -radius, 0));
199 
200  QString text = format_string(text_rect.width(),
201  [&p](const QString& s) -> double { return p.boundingRect(QRectF(), 0, s).width(); });
202 
203  text_size_ = p.boundingRect(QRectF(), 0, text).size();
204 
205  /* Currently, selecting the middle section between two cursors doesn't do
206  * anything, so don't highlight it when selected
207  if (selected()) {
208  p.setBrush(Qt::transparent);
209  p.setPen(highlight_pen());
210  p.drawRoundedRect(delta_rect, radius, radius);
211  } */
212 
213  p.setBrush(hover ? Cursor::FillColor.lighter() : Cursor::FillColor);
214  p.setPen(Cursor::FillColor.darker());
215  p.drawRoundedRect(delta_rect, radius, radius);
216 
217  delta_rect.adjust(1, 1, -1, -1);
218  p.setPen(Cursor::FillColor.lighter());
219  const int highlight_radius = delta_rect.height() / 2 - 2;
220  p.drawRoundedRect(delta_rect, highlight_radius, highlight_radius);
221  label_area_ = delta_rect;
222 
223  p.setPen(text_color);
224  p.drawText(text_rect, Qt::AlignCenter | Qt::AlignVCenter, text);
225 }
226 
228 {
229  if (!enabled())
230  return;
231 
232  p.setPen(Qt::NoPen);
233  p.setBrush(fill_color_);
234 
235  const pair<float, float> offsets(get_cursor_offsets());
236  const int l = (int)max(min(offsets.first, offsets.second), 0.0f);
237  const int r = (int)min(max(offsets.first, offsets.second), (float)pp.width());
238 
239  p.drawRect(l, pp.top(), r - l, pp.height());
240 }
241 
242 QString CursorPair::format_string(int max_width, std::function<double(const QString&)> query_size)
243 {
244  int time_precision = 12;
245  int freq_precision = 12;
246 
247  QString s = format_string_sub(time_precision, freq_precision);
248 
249  // Try full "{time} s / {freq} Hz" format
250  if ((max_width <= 0) || (query_size(s) <= max_width)) {
251  label_incomplete_ = false;
252  return s;
253  }
254 
255  label_incomplete_ = true;
256 
257  // Gradually reduce time precision to match frequency precision
258  while (time_precision > freq_precision) {
259  time_precision--;
260 
261  s = format_string_sub(time_precision, freq_precision);
262  if (query_size(s) <= max_width)
263  return s;
264  }
265 
266  // Gradually reduce both precisions down to zero
267  while (time_precision > 0) {
268  time_precision--;
269  freq_precision--;
270 
271  s = format_string_sub(time_precision, freq_precision);
272  if (query_size(s) <= max_width)
273  return s;
274  }
275 
276  // Try no trailing digits and drop the unit to at least display something
277  s = format_string_sub(0, 0, false);
278 
279  if (query_size(s) <= max_width)
280  return s;
281 
282  // Give up
283  return "...";
284 }
285 
286 pair<float, float> CursorPair::get_cursor_offsets() const
287 {
288  assert(first_);
289  assert(second_);
290 
291  return pair<float, float>(first_->get_x(), second_->get_x());
292 }
293 
294 void CursorPair::on_setting_changed(const QString &key, const QVariant &value)
295 {
297  fill_color_ = QColor::fromRgba(value.value<uint32_t>());
298 
300  show_frequency_ = value.value<bool>();
301 
303  show_interval_ = value.value<bool>();
304 
306  show_samples_ = value.value<bool>();
307 }
308 
309 void CursorPair::on_hover_point_changed(const QWidget* widget, const QPoint& hp)
310 {
311  if (widget != view_.ruler())
312  return;
313 
314  if (!label_incomplete_)
315  return;
316 
317  if (label_area_.contains(hp))
318  QToolTip::showText(view_.mapToGlobal(hp), format_string());
319  else
320  QToolTip::hideText(); // TODO Will break other tooltips when there can be others
321 }
322 
323 QString CursorPair::format_string_sub(int time_precision, int freq_precision, bool show_unit)
324 {
325  QString s = " ";
326 
327  const pv::util::SIPrefix prefix = view_.tick_prefix();
328  const pv::util::Timestamp diff = abs(second_->time() - first_->time());
329 
330  const QString time = Ruler::format_time_with_distance(
331  diff, diff, prefix, (show_unit ? view_.time_unit() : pv::util::TimeUnit::None),
332  time_precision, false);
333 
334  // We can only show a frequency when there's a time base
336  int items = 0;
337 
338  if (show_frequency_) {
339  const QString freq = util::format_value_si(
340  1 / diff.convert_to<double>(), pv::util::SIPrefix::unspecified,
341  freq_precision, (show_unit ? "Hz" : nullptr), false);
342  s = QString("%1").arg(freq);
343  items++;
344  }
345 
346  if (show_interval_) {
347  if (items > 0)
348  s = QString("%1 / %2").arg(s, time);
349  else
350  s = QString("%1").arg(time);
351  items++;
352  }
353 
354  if (show_samples_) {
355  const QString samples = QString::number(
356  (diff * view_.session().get_samplerate()).convert_to<uint64_t>());
357  if (items > 0)
358  s = QString("%1 / %2").arg(s, samples);
359  else
360  s = QString("%1").arg(samples);
361  }
362  } else
363  // In this case, we return the number of samples, really
364  s = time;
365 
366  return s;
367 }
368 
369 } // namespace trace
370 } // namespace views
371 } // namespace pv
Session & session()
Definition: view.cpp:320
const Ruler * ruler() const
Definition: view.cpp:491
util::TimeUnit time_unit() const
Definition: view.cpp:730
pv::util::SIPrefix tick_prefix() const
Definition: view.cpp:686
SIPrefix
Definition: util.hpp:50
void paint_back(QPainter &p, ViewItemPaintParams &pp) override
Definition: cursorpair.cpp:227
T value(details::expression_node< T > *n)
Definition: exprtk.hpp:12358
QString format_value_si(double v, SIPrefix prefix, unsigned precision, QString unit, bool sign)
Definition: util.cpp:153
pair< float, float > get_cursor_offsets() const
Definition: cursorpair.cpp:286
static const QString Key_View_CursorShowSamples
QString format_string(int max_width=0, std::function< double(const QString &)> query_size=[](const QString &s) -> double{(void) s;return 0;})
Definition: cursorpair.cpp:242
QRectF label_rect(const QRectF &rect) const override
Definition: cursorpair.cpp:167
virtual void hover_point_changed(const QPoint &hp)
Definition: viewitem.cpp:151
T max(const T v0, const T v1)
Definition: exprtk.hpp:1411
static const QString Key_View_CursorFillColor
static const int DeltaPadding
Definition: cursorpair.hpp:48
virtual void on_setting_changed(const QString &key, const QVariant &value) override
Definition: cursorpair.cpp:294
static void remove_change_handler(GlobalSettingsInterface *cb)
static const QSizeF LabelPadding
Definition: viewitem.hpp:51
double get_samplerate() const
Definition: session.cpp:910
shared_ptr< Cursor > first() const
Definition: cursorpair.cpp:78
void on_hover_point_changed(const QWidget *widget, const QPoint &hp)
Definition: cursorpair.cpp:309
void setValue(const QString &key, const QVariant &value)
T min(const T v0, const T v1)
Definition: exprtk.hpp:1404
void paint_label(QPainter &p, const QRect &rect, bool hover) override
Definition: cursorpair.cpp:185
bool cursors_shown() const
Definition: view.cpp:968
static const QString Key_View_CursorShowFrequency
QString format_string_sub(int time_precision, int freq_precision, bool show_unit=true)
Definition: cursorpair.cpp:323
shared_ptr< Cursor > first_
Definition: cursorpair.hpp:125
static QString format_time_with_distance(const pv::util::Timestamp &distance, const pv::util::Timestamp &t, pv::util::SIPrefix prefix=pv::util::SIPrefix::unspecified, pv::util::TimeUnit unit=pv::util::TimeUnit::Time, unsigned precision=0, bool sign=true)
Definition: ruler.cpp:84
static QColor select_text_color(QColor background)
Definition: viewitem.cpp:146
float get_x() const override
Definition: cursorpair.cpp:100
bool enabled() const override
Definition: cursorpair.cpp:73
static const QString Key_View_CursorShowInterval
QMenu * create_header_context_menu(QWidget *parent) override
Definition: cursorpair.cpp:124
QPoint drag_point(const QRect &rect) const override
Definition: cursorpair.cpp:113
boost::multiprecision::number< boost::multiprecision::cpp_dec_float< 24 >, boost::multiprecision::et_off > Timestamp
Timestamp type providing yoctosecond resolution.
Definition: util.hpp:67
static const QColor FillColor
Definition: cursor.hpp:42
static const int ArrowSize
Definition: timemarker.hpp:53
shared_ptr< Cursor > second_
Definition: cursorpair.hpp:125
shared_ptr< Cursor > second() const
Definition: cursorpair.cpp:83
void set_time(const pv::util::Timestamp &time) override
Definition: cursorpair.cpp:88
virtual const pv::util::Timestamp time() const override
Definition: cursorpair.cpp:95
pv::widgets::Popup * create_popup(QWidget *parent) override
Definition: cursorpair.cpp:118
static void add_change_handler(GlobalSettingsInterface *cb)
virtual const pv::util::Timestamp delta(const pv::util::Timestamp &other) const override
Definition: cursorpair.cpp:105