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
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
|
/****************************************************************************
**
** Copyright (C) 2022 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:COMM$
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** $QT_END_LICENSE$
**
**
**
**
**
**
**
**
**
**
**
**
**
**
**
**
**
**
**
******************************************************************************/
#include <qaudio.h>
#include <qmath.h>
#include <QDebug>
QT_BEGIN_NAMESPACE
#define LOG100 4.60517018599
/*!
\namespace QAudio
\ingroup multimedia-namespaces
\brief The QAudio namespace contains enums used by the audio classes.
\inmodule QtMultimedia
\ingroup multimedia
\ingroup multimedia_audio
*/
/*!
\enum QAudio::Error
\value NoError No errors have occurred
\value OpenError An error occurred opening the audio device
\value IOError An error occurred during read/write of audio device
\value UnderrunError Audio data is not being fed to the audio device at a fast enough rate
\value FatalError A non-recoverable error has occurred, the audio device is not usable at this time.
*/
/*!
\enum QAudio::State
\value ActiveState Audio data is being processed, this state is set after start() is called
and while audio data is available to be processed.
\value SuspendedState The audio stream is in a suspended state. Entered after suspend() is called
or when another stream takes control of the audio device. In the later case,
a call to resume will return control of the audio device to this stream. This
should usually only be done upon user request.
\value StoppedState The audio device is closed, and is not processing any audio data
\value IdleState The QIODevice passed in has no data and audio system's buffer is empty, this state
is set after start() is called and while no audio data is available to be processed.
*/
/*!
\enum QAudio::VolumeScale
This enum defines the different audio volume scales.
\value LinearVolumeScale Linear scale. \c 0.0 (0%) is silence and \c 1.0 (100%) is full
volume. All Qt Multimedia classes that have an audio volume use
a linear scale.
\value CubicVolumeScale Cubic scale. \c 0.0 (0%) is silence and \c 1.0 (100%) is full
volume.
\value LogarithmicVolumeScale Logarithmic Scale. \c 0.0 (0%) is silence and \c 1.0 (100%) is
full volume. UI volume controls should usually use a logarithmic
scale.
\value DecibelVolumeScale Decibel (dB, amplitude) logarithmic scale. \c -200 is silence
and \c 0 is full volume.
\since 5.8
\sa QAudio::convertVolume()
*/
namespace QAudio
{
/*!
Converts an audio \a volume \a from a volume scale \a to another, and returns the result.
Depending on the context, different scales are used to represent audio volume. All Qt Multimedia
classes that have an audio volume use a linear scale, the reason is that the loudness of a
speaker is controlled by modulating its voltage on a linear scale. The human ear on the other
hand, perceives loudness in a logarithmic way. Using a logarithmic scale for volume controls
is therefore appropriate in most applications. The decibel scale is logarithmic by nature and
is commonly used to define sound levels, it is usually used for UI volume controls in
professional audio applications. The cubic scale is a computationally cheap approximation of a
logarithmic scale, it provides more control over lower volume levels.
The following example shows how to convert the volume value from a slider control before passing
it to a QMediaPlayer. As a result, the perceived increase in volume is the same when increasing
the volume slider from 20 to 30 as it is from 50 to 60:
\snippet multimedia-snippets/audio.cpp Volume conversion
\since 5.8
\sa VolumeScale, QAudioSink::setVolume(), QAudioSource::setVolume(),
QSoundEffect::setVolume()
*/
float convertVolume(float volume, VolumeScale from, VolumeScale to)
{
switch (from) {
case LinearVolumeScale:
volume = qMax(float(0), volume);
switch (to) {
case LinearVolumeScale:
return volume;
case CubicVolumeScale:
return qPow(volume, float(1 / 3.0));
case LogarithmicVolumeScale:
return 1 - std::exp(-volume * LOG100);
case DecibelVolumeScale:
if (volume < 0.001)
return float(-200);
else
return float(20.0) * std::log10(volume);
}
break;
case CubicVolumeScale:
volume = qMax(float(0), volume);
switch (to) {
case LinearVolumeScale:
return volume * volume * volume;
case CubicVolumeScale:
return volume;
case LogarithmicVolumeScale:
return 1 - std::exp(-volume * volume * volume * LOG100);
case DecibelVolumeScale:
if (volume < 0.001)
return float(-200);
else
return float(3.0 * 20.0) * std::log10(volume);
}
break;
case LogarithmicVolumeScale:
volume = qMax(float(0), volume);
switch (to) {
case LinearVolumeScale:
if (volume > 0.99)
return 1;
else
return -std::log(1 - volume) / LOG100;
case CubicVolumeScale:
if (volume > 0.99)
return 1;
else
return qPow(-std::log(1 - volume) / LOG100, float(1 / 3.0));
case LogarithmicVolumeScale:
return volume;
case DecibelVolumeScale:
if (volume < 0.001)
return float(-200);
else if (volume > 0.99)
return 0;
else
return float(20.0) * std::log10(-std::log(1 - volume) / LOG100);
}
break;
case DecibelVolumeScale:
switch (to) {
case LinearVolumeScale:
return qPow(float(10.0), volume / float(20.0));
case CubicVolumeScale:
return qPow(float(10.0), volume / float(3.0 * 20.0));
case LogarithmicVolumeScale:
if (qFuzzyIsNull(volume))
return 1;
else
return 1 - std::exp(-qPow(float(10.0), volume / float(20.0)) * LOG100);
case DecibelVolumeScale:
return volume;
}
break;
}
return volume;
}
}
#ifndef QT_NO_DEBUG_STREAM
QDebug operator<<(QDebug dbg, QAudio::Error error)
{
QDebugStateSaver saver(dbg);
dbg.nospace();
switch (error) {
case QAudio::NoError:
dbg << "NoError";
break;
case QAudio::OpenError:
dbg << "OpenError";
break;
case QAudio::IOError:
dbg << "IOError";
break;
case QAudio::UnderrunError:
dbg << "UnderrunError";
break;
case QAudio::FatalError:
dbg << "FatalError";
break;
}
return dbg;
}
QDebug operator<<(QDebug dbg, QAudio::State state)
{
QDebugStateSaver saver(dbg);
dbg.nospace();
switch (state) {
case QAudio::ActiveState:
dbg << "ActiveState";
break;
case QAudio::SuspendedState:
dbg << "SuspendedState";
break;
case QAudio::StoppedState:
dbg << "StoppedState";
break;
case QAudio::IdleState:
dbg << "IdleState";
break;
}
return dbg;
}
QDebug operator<<(QDebug dbg, QAudio::VolumeScale scale)
{
QDebugStateSaver saver(dbg);
dbg.nospace();
switch (scale) {
case QAudio::LinearVolumeScale:
dbg << "LinearVolumeScale";
break;
case QAudio::CubicVolumeScale:
dbg << "CubicVolumeScale";
break;
case QAudio::LogarithmicVolumeScale:
dbg << "LogarithmicVolumeScale";
break;
case QAudio::DecibelVolumeScale:
dbg << "DecibelVolumeScale";
break;
}
return dbg;
}
#endif
QT_END_NAMESPACE
|