]> sigrok.org Git - pulseview.git/blob - pv/prop/int.cpp
Fix #1292/1294 by snapping to any edge when not hovering over a signal
[pulseview.git] / pv / prop / int.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, see <http://www.gnu.org/licenses/>.
18  */
19
20 #include <cassert>
21 #include <cstdint>
22
23 #include <QSpinBox>
24
25 #include "int.hpp"
26
27 using boost::optional;
28 using std::max;
29 using std::min;
30 using std::pair;
31
32 namespace pv {
33 namespace prop {
34
35 Int::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
48 QWidget* 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
112 void 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
148 void 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
185 void Int::on_value_changed(int)
186 {
187         commit();
188 }
189
190 }  // namespace prop
191 }  // namespace pv