]> sigrok.org Git - pulseview.git/blame_incremental - pv/popups/probes.cpp
Update for 'probe_group' -> 'channel_group' rename in libsigrok.
[pulseview.git] / pv / popups / probes.cpp
... / ...
CommitLineData
1/*
2 * This file is part of the PulseView project.
3 *
4 * Copyright (C) 2012 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 <map>
22
23#include <boost/foreach.hpp>
24
25#include <QCheckBox>
26#include <QFormLayout>
27#include <QGridLayout>
28#include <QLabel>
29
30#include "probes.h"
31
32#include <pv/device/devinst.h>
33#include <pv/prop/binding/deviceoptions.h>
34#include <pv/sigsession.h>
35#include <pv/view/signal.h>
36
37using namespace Qt;
38
39using boost::shared_ptr;
40using std::map;
41using std::set;
42using std::vector;
43
44using pv::view::Signal;
45
46namespace pv {
47namespace popups {
48
49Probes::Probes(SigSession &session, QWidget *parent) :
50 Popup(parent),
51 _session(session),
52 _updating_probes(false),
53 _enable_all_probes(tr("Enable All"), this),
54 _disable_all_probes(tr("Disable All"), this),
55 _check_box_mapper(this)
56{
57 // Create the layout
58 setLayout(&_layout);
59
60 shared_ptr<device::DevInst> dev_inst = _session.get_device();
61 assert(dev_inst);
62 const sr_dev_inst *const sdi = dev_inst->dev_inst();
63 assert(sdi);
64
65 // Collect a set of signals
66 map<const sr_probe*, shared_ptr<Signal> > signal_map;
67 const vector< shared_ptr<Signal> > sigs = _session.get_signals();
68 BOOST_FOREACH(const shared_ptr<Signal> &sig, sigs)
69 signal_map[sig->probe()] = sig;
70
71 // Populate channel groups
72 for (const GSList *g = sdi->channel_groups; g; g = g->next)
73 {
74 const sr_channel_group *const group =
75 (const sr_channel_group*)g->data;
76 assert(group);
77
78 // Make a set of signals, and removed this signals from the
79 // signal map.
80 vector< shared_ptr<Signal> > group_sigs;
81 for (const GSList *p = group->probes; p; p = p->next)
82 {
83 const sr_probe *const probe = (const sr_probe*)p->data;
84 assert(probe);
85
86 const map<const sr_probe*, shared_ptr<Signal> >::
87 iterator iter = signal_map.find(probe);
88 assert(iter != signal_map.end());
89
90 group_sigs.push_back((*iter).second);
91 signal_map.erase(iter);
92 }
93
94 populate_group(group, group_sigs);
95 }
96
97 // Make a vector of the remaining probes
98 vector< shared_ptr<Signal> > global_sigs;
99 for (const GSList *p = sdi->probes; p; p = p->next)
100 {
101 const sr_probe *const probe = (const sr_probe*)p->data;
102 assert(probe);
103
104 const map<const sr_probe*, shared_ptr<Signal> >::
105 const_iterator iter = signal_map.find(probe);
106 if (iter != signal_map.end())
107 global_sigs.push_back((*iter).second);
108 }
109
110 // Create a group
111 populate_group(NULL, global_sigs);
112
113 // Create the enable/disable all buttons
114 connect(&_enable_all_probes, SIGNAL(clicked()),
115 this, SLOT(enable_all_probes()));
116 connect(&_disable_all_probes, SIGNAL(clicked()),
117 this, SLOT(disable_all_probes()));
118
119 _enable_all_probes.setFlat(true);
120 _disable_all_probes.setFlat(true);
121
122 _buttons_bar.addWidget(&_enable_all_probes);
123 _buttons_bar.addWidget(&_disable_all_probes);
124 _buttons_bar.addStretch(1);
125
126 _layout.addRow(&_buttons_bar);
127
128 // Connect the check-box signal mapper
129 connect(&_check_box_mapper, SIGNAL(mapped(QWidget*)),
130 this, SLOT(on_probe_checked(QWidget*)));
131}
132
133void Probes::set_all_probes(bool set)
134{
135 _updating_probes = true;
136
137 for (map<QCheckBox*, shared_ptr<Signal> >::const_iterator i =
138 _check_box_signal_map.begin();
139 i != _check_box_signal_map.end(); i++)
140 {
141 const shared_ptr<Signal> sig = (*i).second;
142 assert(sig);
143
144 sig->enable(set);
145 (*i).first->setChecked(set);
146 }
147
148 _updating_probes = false;
149}
150
151void Probes::populate_group(const sr_channel_group *group,
152 const vector< shared_ptr<pv::view::Signal> > sigs)
153{
154 using pv::prop::binding::DeviceOptions;
155
156 // Only bind options if this is a group. We don't do it for general
157 // options, because these properties are shown in the device config
158 // popup.
159 shared_ptr<DeviceOptions> binding;
160 if (group)
161 binding = shared_ptr<DeviceOptions>(new DeviceOptions(
162 _session.get_device(), group));
163
164 // Create a title if the group is going to have any content
165 if ((!sigs.empty() || (binding && !binding->properties().empty())) &&
166 group && group->name)
167 _layout.addRow(new QLabel(
168 QString("<h3>%1</h3>").arg(group->name)));
169
170 // Create the channel group grid
171 QGridLayout *const probe_grid =
172 create_channel_group_grid(sigs);
173 _layout.addRow(probe_grid);
174
175 // Create the channel group options
176 if (binding)
177 {
178 binding->add_properties_to_form(&_layout, true);
179 _group_bindings.push_back(binding);
180 }
181}
182
183QGridLayout* Probes::create_channel_group_grid(
184 const vector< shared_ptr<pv::view::Signal> > sigs)
185{
186 int row = 0, col = 0;
187 QGridLayout *const grid = new QGridLayout();
188
189 BOOST_FOREACH(const shared_ptr<pv::view::Signal>& sig, sigs)
190 {
191 assert(sig);
192
193 QCheckBox *const checkbox = new QCheckBox(sig->get_name());
194 _check_box_mapper.setMapping(checkbox, checkbox);
195 connect(checkbox, SIGNAL(toggled(bool)),
196 &_check_box_mapper, SLOT(map()));
197
198 grid->addWidget(checkbox, row, col);
199
200 _check_box_signal_map[checkbox] = sig;
201
202 if(++col >= 8)
203 col = 0, row++;
204 }
205
206 return grid;
207}
208
209void Probes::showEvent(QShowEvent *e)
210{
211 pv::widgets::Popup::showEvent(e);
212
213 _updating_probes = true;
214
215 for (map<QCheckBox*, shared_ptr<Signal> >::const_iterator i =
216 _check_box_signal_map.begin();
217 i != _check_box_signal_map.end(); i++)
218 {
219 const shared_ptr<Signal> sig = (*i).second;
220 assert(sig);
221
222 (*i).first->setChecked(sig->enabled());
223 }
224
225 _updating_probes = false;
226}
227
228void Probes::on_probe_checked(QWidget *widget)
229{
230 if (_updating_probes)
231 return;
232
233 QCheckBox *const check_box = (QCheckBox*)widget;
234 assert(check_box);
235
236 // Look up the signal of this check-box
237 map< QCheckBox*, shared_ptr<Signal> >::const_iterator iter =
238 _check_box_signal_map.find((QCheckBox*)check_box);
239 assert(iter != _check_box_signal_map.end());
240
241 const shared_ptr<pv::view::Signal> s = (*iter).second;
242 assert(s);
243
244 s->enable(check_box->isChecked());
245}
246
247void Probes::enable_all_probes()
248{
249 set_all_probes(true);
250}
251
252void Probes::disable_all_probes()
253{
254 set_all_probes(false);
255}
256
257} // popups
258} // pv