]> sigrok.org Git - pulseview.git/blame_incremental - pv/prop/int.cpp
Fix #1292/1294 by snapping to any edge when not hovering over a signal
[pulseview.git] / pv / prop / int.cpp
... / ...
CommitLineData
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 <cassert>
21#include <cstdint>
22
23#include <QSpinBox>
24
25#include "int.hpp"
26
27using boost::optional;
28using std::max;
29using std::min;
30using std::pair;
31
32namespace pv {
33namespace prop {
34
35Int::Int(QString name,
36 QString desc,
37 QString suffix,
38 optional< pair<int64_t, int64_t> > range,
39 Getter getter,
40 Setter setter) :
41 Property(name, desc, getter, setter),
42 suffix_(suffix),
43 range_(range),
44 spin_box_(nullptr)
45{
46}
47
48QWidget* Int::get_widget(QWidget *parent, bool auto_commit)
49{
50 int64_t range_min = 0;
51 uint64_t range_max = 0;
52
53 if (spin_box_)
54 return spin_box_;
55
56 if (!getter_)
57 return nullptr;
58
59 value_ = getter_();
60
61 GVariant *value = value_.gobj();
62 if (!value)
63 return nullptr;
64
65 spin_box_ = new QSpinBox(parent);
66 spin_box_->setSuffix(suffix_);
67
68 const GVariantType *const type = g_variant_get_type(value);
69 assert(type);
70
71 if (g_variant_type_equal(type, G_VARIANT_TYPE_BYTE)) {
72 range_min = 0, range_max = UINT8_MAX;
73 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT16)) {
74 range_min = INT16_MIN, range_max = INT16_MAX;
75 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT16)) {
76 range_min = 0, range_max = UINT16_MAX;
77 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT32)) {
78 range_min = INT32_MIN, range_max = INT32_MAX;
79 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT32)) {
80 range_min = 0, range_max = UINT32_MAX;
81 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT64)) {
82 range_min = INT64_MIN, range_max = INT64_MAX;
83 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT64)) {
84 range_min = 0, range_max = UINT64_MAX;
85 } else {
86 // Unexpected value type.
87 assert(false);
88 }
89
90 // @todo sigrok supports 64-bit quantities, but Qt does not have a
91 // standard widget to allow the values to be modified over the full
92 // 64-bit range on 32-bit machines. To solve the issue we need a
93 // custom widget.
94
95 range_min = max(range_min, (int64_t)INT_MIN);
96 range_max = min(range_max, (uint64_t)INT_MAX);
97
98 if (range_)
99 spin_box_->setRange((int)range_->first, (int)range_->second);
100 else
101 spin_box_->setRange((int)range_min, (int)range_max);
102
103 update_widget();
104
105 if (auto_commit)
106 connect(spin_box_, SIGNAL(valueChanged(int)),
107 this, SLOT(on_value_changed(int)));
108
109 return spin_box_;
110}
111
112void Int::update_widget()
113{
114 if (!spin_box_)
115 return;
116
117 value_ = getter_();
118 GVariant *value = value_.gobj();
119 assert(value);
120
121 const GVariantType *const type = g_variant_get_type(value);
122 assert(type);
123
124 int64_t int_val = 0;
125
126 if (g_variant_type_equal(type, G_VARIANT_TYPE_BYTE)) {
127 int_val = g_variant_get_byte(value);
128 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT16)) {
129 int_val = g_variant_get_int16(value);
130 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT16)) {
131 int_val = g_variant_get_uint16(value);
132 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT32)) {
133 int_val = g_variant_get_int32(value);
134 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT32)) {
135 int_val = g_variant_get_uint32(value);
136 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT64)) {
137 int_val = g_variant_get_int64(value);
138 } else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT64)) {
139 int_val = g_variant_get_uint64(value);
140 } else {
141 // Unexpected value type.
142 assert(false);
143 }
144
145 spin_box_->setValue((int)int_val);
146}
147
148void Int::commit()
149{
150 assert(setter_);
151
152 if (!spin_box_)
153 return;
154
155 GVariant *new_value = nullptr;
156 const GVariantType *const type = g_variant_get_type(value_.gobj());
157 assert(type);
158
159 if (g_variant_type_equal(type, G_VARIANT_TYPE_BYTE))
160 new_value = g_variant_new_byte(spin_box_->value());
161 else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT16))
162 new_value = g_variant_new_int16(spin_box_->value());
163 else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT16))
164 new_value = g_variant_new_uint16(spin_box_->value());
165 else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT32))
166 new_value = g_variant_new_int32(spin_box_->value());
167 else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT32))
168 new_value = g_variant_new_uint32(spin_box_->value());
169 else if (g_variant_type_equal(type, G_VARIANT_TYPE_INT64))
170 new_value = g_variant_new_int64(spin_box_->value());
171 else if (g_variant_type_equal(type, G_VARIANT_TYPE_UINT64))
172 new_value = g_variant_new_uint64(spin_box_->value());
173 else {
174 // Unexpected value type.
175 assert(false);
176 }
177
178 assert(new_value);
179
180 value_ = Glib::VariantBase(new_value);
181
182 setter_(value_);
183}
184
185void Int::on_value_changed(int)
186{
187 commit();
188}
189
190} // namespace prop
191} // namespace pv