-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
/
Copy pathcontrol.h
237 lines (188 loc) · 7.82 KB
/
control.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
#pragma once
#include <QAtomicPointer>
#include <QHash>
#include <QObject>
#include <QSharedPointer>
#include <QString>
#include "control/controlbehavior.h"
#include "control/controlvalue.h"
#include "preferences/usersettings.h"
class ControlObject;
enum class ControlFlag {
None = 0,
/// Do not throw an assertion if the key is invalid. Needed for controller
/// mappings and skins.
AllowInvalidKey = 1,
/// Don't throw an assertion when trying to access a non-existing CO.
/// Needed for controller mappings and skins.
NoAssertIfMissing = 1 << 1,
/// Don't log a warning when trying to access a non-existing CO.
NoWarnIfMissing = (1 << 2) | NoAssertIfMissing,
AllowMissingOrInvalid = AllowInvalidKey | NoAssertIfMissing,
};
Q_DECLARE_FLAGS(ControlFlags, ControlFlag)
Q_DECLARE_OPERATORS_FOR_FLAGS(ControlFlags)
class ControlDoublePrivate : public QObject {
Q_OBJECT
public:
~ControlDoublePrivate() override;
// TODO: don't expose this implementation detail
constexpr static double kDefaultValue = 0.0;
// Used to implement control persistence. All controls that are marked
// "persist in user config" get and set their value on creation/deletion
// using this UserSettings.
static void setUserConfig(const UserSettingsPointer& pConfig);
// Adds a ConfigKey for 'alias' to the control for 'key'. Can be used for
// supporting a legacy / deprecated control. The 'key' control must exist
// for this to work.
static void insertAlias(const ConfigKey& alias, const ConfigKey& key);
// Gets the ControlDoublePrivate matching the given ConfigKey. If pCreatorCO
// is non-NULL, allocates a new ControlDoublePrivate for the ConfigKey if
// one does not exist.
static QSharedPointer<ControlDoublePrivate> getControl(
const ConfigKey& key,
ControlFlags flags = ControlFlag::None,
ControlObject* pCreatorCO = nullptr,
bool bIgnoreNops = true,
bool bTrack = false,
bool bPersist = false,
double defaultValue = kDefaultValue);
static QSharedPointer<ControlDoublePrivate> getDefaultControl();
// Returns a list of all existing instances.
static QList<QSharedPointer<ControlDoublePrivate>> getAllInstances();
// Clears all existing instances and returns them as a list.
static QList<QSharedPointer<ControlDoublePrivate>> takeAllInstances();
static QHash<ConfigKey, ConfigKey> getControlAliases();
const QString& name() const {
return m_name;
}
void setName(const QString& name) {
m_name = name;
}
const QString& description() const {
return m_description;
}
void setDescription(const QString& description) {
m_description = description;
}
void setKbdRepeatable(bool enable) {
m_kbdRepeatable = enable;
}
bool getKbdRepeatable() const {
return m_kbdRepeatable;
}
// Sets the control value.
void set(double value, QObject* pSender);
// directly sets the control value. Must be used from and only from the
// ValueChangeRequest slot.
void setAndConfirm(double value, QObject* pSender);
// Gets the control value.
double get() const {
return m_value.getValue();
}
// Resets the control value to its default.
void reset();
// Set the behavior to be used when setting values and translating between
// parameter and value space. Returns the previously set behavior (if any).
// Callers must allocate the passed behavior using new and ownership to this
// memory is passed with the function call!!
// TODO: Pass a std::unique_ptr instead of a plain pointer to ensure this
// transfer of ownership.
void setBehavior(ControlNumericBehavior* pBehavior);
void setParameter(double dParam, QObject* pSender);
double getParameter() const;
double getParameterForValue(double value) const;
double getParameterForMidi(double midiValue) const;
void setValueFromMidi(MidiOpCode opcode, double dParam);
double getMidiParameter() const;
bool ignoreNops() const {
return m_bIgnoreNops;
}
void setDefaultValue(double dValue) {
m_defaultValue.setValue(dValue);
}
double defaultValue() const {
return m_defaultValue.getValue();
}
ControlObject* getCreatorCO() const {
return m_pCreatorCO.loadAcquire();
}
bool resetCreatorCO(ControlObject* pCreatorCO) {
return m_pCreatorCO.testAndSetOrdered(pCreatorCO, nullptr);
}
void deleteCreatorCO();
const ConfigKey& getKey() {
return m_key;
}
// Connects a slot to the ValueChange request for CO validation. All change
// requests issued by set are routed though the connected slot. This can
// decide with its own thread safe solution if the requested value can be
// confirmed by setAndConfirm() or not. Note: Once connected, the CO value
// itself is ONLY set by setAndConfirm() typically called in the connected
// slot.
template <typename Receiver, typename Slot>
bool connectValueChangeRequest(Receiver receiver,
Slot func, Qt::ConnectionType type) {
// confirmation is only required if connect was successful
m_confirmRequired = connect(this, &ControlDoublePrivate::valueChangeRequest,
receiver, func, type);
return m_confirmRequired;
}
signals:
// Emitted when the ControlDoublePrivate value changes. pSender is a
// pointer to the setter of the value (potentially NULL).
void valueChanged(double value, QObject* pSender);
void valueChangeRequest(double value);
protected:
ControlDoublePrivate();
private:
ControlDoublePrivate(
const ConfigKey& key,
ControlObject* pCreatorCO,
bool bIgnoreNops,
bool bTrack,
bool bPersist,
double defaultValue,
bool confirmRequired);
ControlDoublePrivate(ControlDoublePrivate&&) = delete;
ControlDoublePrivate(const ControlDoublePrivate&) = delete;
ControlDoublePrivate& operator=(ControlDoublePrivate&&) = delete;
ControlDoublePrivate& operator=(const ControlDoublePrivate&) = delete;
void initialize(double defaultValue);
virtual void setInner(double value, QObject* pSender);
const ConfigKey m_key;
QSharedPointer<ControlNumericBehavior> m_pBehavior;
// User-visible, i18n name for what the control is.
QString m_name;
// User-visible, i18n description for what the control does.
QString m_description;
// The control value.
ControlValueAtomic<double> m_value;
// The default control value.
ControlValueAtomic<double> m_defaultValue;
QAtomicPointer<ControlObject> m_pCreatorCO;
// name of the key to track using stats framework, unless the m_trackingKey isNull().
QString m_trackingKey;
// Note: keep the order of the members below to not introduce gaps due to
// memory alignment in this often used class.
bool m_confirmRequired;
// Whether the control should persist in the Mixxx user configuration. The
// value is loaded from configuration when the control is created and
// written to the configuration when the control is deleted.
bool m_bPersistInConfiguration;
// Whether to ignore sets which would have no effect.
bool m_bIgnoreNops;
// If true, this control will be issued repeatedly if the keyboard key is held.
bool m_kbdRepeatable;
};
/// The constant ControlDoublePrivate version is used as dummy for default
/// constructed control objects
class ControlDoublePrivateConst : public ControlDoublePrivate {
Q_OBJECT
public:
~ControlDoublePrivateConst() override = default;
private:
void setInner(double, QObject*) override {
DEBUG_ASSERT(!"Trying to modify a default constructed (const) control object");
};
};