]> sigrok.org Git - pulseview.git/blob - pv/view/cursorpair.cpp
CursorPair: Draw highlight when selected
[pulseview.git] / pv / view / cursorpair.cpp
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, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
19  */
20
21 #include "cursorpair.hpp"
22
23 #include "view.hpp"
24 #include "pv/util.hpp"
25
26 #include <cassert>
27 #include <algorithm>
28
29 using std::max;
30 using std::make_pair;
31 using std::min;
32 using std::shared_ptr;
33 using std::pair;
34
35 namespace pv {
36 namespace view {
37
38 const int CursorPair::DeltaPadding = 8;
39
40 CursorPair::CursorPair(View &view) :
41         TimeItem(view),
42         first_(new Cursor(view, 0.0)),
43         second_(new Cursor(view, 1.0))
44 {
45 }
46
47 bool CursorPair::enabled() const
48 {
49         return view_.cursors_shown();
50 }
51
52 shared_ptr<Cursor> CursorPair::first() const
53 {
54         return first_;
55 }
56
57 shared_ptr<Cursor> CursorPair::second() const
58 {
59         return second_;
60 }
61
62 void CursorPair::set_time(double time) {
63         const double delta = second_->time() - first_->time();
64         first_->set_time(time);
65         second_->set_time(time + delta);
66 }
67
68 float CursorPair::get_x() const
69 {
70         return (first_->get_x() + second_->get_x()) / 2.0f;
71 }
72
73 QPoint CursorPair::point() const
74 {
75         return first_->point();
76 }
77
78 pv::widgets::Popup* CursorPair::create_popup(QWidget *parent)
79 {
80         (void)parent;
81         return nullptr;
82 }
83
84 QRectF CursorPair::label_rect(const QRectF &rect) const
85 {
86         const QSizeF label_size(
87                 text_size_.width() + View::LabelPadding.width() * 2,
88                 text_size_.height() + View::LabelPadding.height() * 2);
89         const pair<float, float> offsets(get_cursor_offsets());
90         const pair<float, float> normal_offsets(
91                 (offsets.first < offsets.second) ? offsets :
92                 make_pair(offsets.second, offsets.first));
93
94         const float height = label_size.height();
95         const float left = max(normal_offsets.first + DeltaPadding, -height);
96         const float right = min(normal_offsets.second - DeltaPadding,
97                 (float)rect.width() + height);
98
99         return QRectF(left, rect.height() - label_size.height() -
100                 TimeMarker::ArrowSize - TimeMarker::Offset - 0.5f,
101                 right - left, height);
102 }
103
104 void CursorPair::draw_markers(QPainter &p, const QRect &rect)
105 {
106         assert(first_);
107         assert(second_);
108
109         const unsigned int prefix = view_.tick_prefix();
110
111         compute_text_size(p, prefix);
112         QRectF delta_rect(label_rect(rect));
113
114         const int radius = delta_rect.height() / 2;
115         const QRectF text_rect(delta_rect.intersected(
116                 rect).adjusted(radius, 0, -radius, 0));
117         if(text_rect.width() >= text_size_.width())
118         {
119                 const int highlight_radius = delta_rect.height() / 2 - 2;
120
121                 if (selected()) {
122                         p.setBrush(Qt::transparent);
123                         p.setPen(highlight_pen());
124                         p.drawRoundedRect(delta_rect, radius, radius);
125                 }
126
127                 p.setBrush(Cursor::FillColour);
128                 p.setPen(Cursor::FillColour.darker());
129                 p.drawRoundedRect(delta_rect, radius, radius);
130
131                 delta_rect.adjust(1, 1, -1, -1);
132                 p.setPen(Cursor::FillColour.lighter());
133                 p.drawRoundedRect(delta_rect, highlight_radius, highlight_radius);
134
135                 p.setPen(SelectableItem::select_text_colour(
136                         Cursor::FillColour));
137                 p.drawText(text_rect, Qt::AlignCenter | Qt::AlignVCenter,
138                         pv::util::format_time(second_->time() - first_->time(), prefix, 2));
139         }
140
141         // Paint the cursor markers
142         first_->paint_label(p, rect);
143         second_->paint_label(p, rect);
144 }
145
146 void CursorPair::draw_viewport_background(QPainter &p,
147         const QRect &rect)
148 {
149         p.setPen(Qt::NoPen);
150         p.setBrush(QBrush(View::CursorAreaColour));
151
152         const pair<float, float> offsets(get_cursor_offsets());
153         const int l = (int)max(min(
154                 offsets.first, offsets.second), 0.0f);
155         const int r = (int)min(max(
156                 offsets.first, offsets.second), (float)rect.width());
157
158         p.drawRect(l, 0, r - l, rect.height());
159 }
160
161 void CursorPair::draw_viewport_foreground(QPainter &p,
162         const QRect &rect)
163 {
164         assert(first_);
165         assert(second_);
166
167         first_->paint(p, rect);
168         second_->paint(p, rect);
169 }
170
171 void CursorPair::compute_text_size(QPainter &p, unsigned int prefix)
172 {
173         assert(first_);
174         assert(second_);
175
176         text_size_ = p.boundingRect(QRectF(), 0, pv::util::format_time(
177                 second_->time() - first_->time(), prefix, 2)).size();
178 }
179
180 pair<float, float> CursorPair::get_cursor_offsets() const
181 {
182         assert(first_);
183         assert(second_);
184
185         return pair<float, float>(
186                 (first_->time() - view_.offset()) / view_.scale(),
187                 (second_->time() - view_.offset()) / view_.scale());
188 }
189
190 } // namespace view
191 } // namespace pv