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
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
|
/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
** 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.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 3 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL3 included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 3 requirements
** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 2.0 or (at your option) the GNU General
** Public license version 3 or any later version approved by the KDE Free
** Qt Foundation. The licenses are as published by the Free Software
** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
** included in the packaging of this file. Please review the following
** information to ensure the GNU General Public License requirements will
** be met: https://www.gnu.org/licenses/gpl-2.0.html and
** https://www.gnu.org/licenses/gpl-3.0.html.
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include <qaudio.h>
#include <qmath.h>
#include <QDebug>
QT_BEGIN_NAMESPACE
#define LOG100 4.60517018599
static void qRegisterAudioMetaTypes()
{
qRegisterMetaType<QAudio::Error>();
qRegisterMetaType<QAudio::State>();
qRegisterMetaType<QAudio::Mode>();
qRegisterMetaType<QAudio::Role>();
qRegisterMetaType<QAudio::VolumeScale>();
}
Q_CONSTRUCTOR_FUNCTION(qRegisterAudioMetaTypes)
/*!
\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 device is in a suspended state, this state will only be entered
after suspend() is called.
\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::Mode
\value AudioOutput audio output device
\value AudioInput audio input device
*/
/*!
\enum QAudio::Role
This enum describes the role of an audio stream.
\value UnknownRole The role is unknown or undefined
\value MusicRole Music
\value VideoRole Soundtrack from a movie or a video
\value VoiceCommunicationRole Voice communications, such as telephony
\value AlarmRole Alarm
\value NotificationRole Notification, such as an incoming e-mail or a chat request
\value RingtoneRole Ringtone
\value AccessibilityRole For accessibility, such as with a screen reader
\value SonificationRole Sonification, such as with user interface sounds
\value GameRole Game audio
\since 5.6
\sa QMediaPlayer::setAudioRole()
*/
/*!
\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
{
/*!
\fn qreal QAudio::convertVolume(qreal volume, VolumeScale from, VolumeScale to)
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, QMediaPlayer::setVolume(), QAudioOutput::setVolume(),
QAudioInput::setVolume(), QSoundEffect::setVolume(), QMediaRecorder::setVolume()
*/
qreal convertVolume(qreal volume, VolumeScale from, VolumeScale to)
{
switch (from) {
case LinearVolumeScale:
volume = qMax(qreal(0), volume);
switch (to) {
case LinearVolumeScale:
return volume;
case CubicVolumeScale:
return qPow(volume, qreal(1 / 3.0));
case LogarithmicVolumeScale:
return 1 - std::exp(-volume * LOG100);
case DecibelVolumeScale:
if (volume < 0.001)
return qreal(-200);
else
return qreal(20.0) * std::log10(volume);
}
break;
case CubicVolumeScale:
volume = qMax(qreal(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 qreal(-200);
else
return qreal(3.0 * 20.0) * std::log10(volume);
}
break;
case LogarithmicVolumeScale:
volume = qMax(qreal(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, qreal(1 / 3.0));
case LogarithmicVolumeScale:
return volume;
case DecibelVolumeScale:
if (volume < 0.001)
return qreal(-200);
else if (volume > 0.99)
return 0;
else
return qreal(20.0) * std::log10(-std::log(1 - volume) / LOG100);
}
break;
case DecibelVolumeScale:
switch (to) {
case LinearVolumeScale:
return qPow(qreal(10.0), volume / qreal(20.0));
case CubicVolumeScale:
return qPow(qreal(10.0), volume / qreal(3.0 * 20.0));
case LogarithmicVolumeScale:
if (qFuzzyIsNull(volume))
return 1;
else
return 1 - std::exp(-qPow(qreal(10.0), volume / qreal(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::Mode mode)
{
QDebugStateSaver saver(dbg);
dbg.nospace();
switch (mode) {
case QAudio::AudioInput:
dbg << "AudioInput";
break;
case QAudio::AudioOutput:
dbg << "AudioOutput";
break;
}
return dbg;
}
QDebug operator<<(QDebug dbg, QAudio::Role role)
{
QDebugStateSaver saver(dbg);
dbg.nospace();
switch (role) {
case QAudio::UnknownRole:
dbg << "UnknownRole";
break;
case QAudio::AccessibilityRole:
dbg << "AccessibilityRole";
break;
case QAudio::AlarmRole:
dbg << "AlarmRole";
break;
case QAudio::GameRole:
dbg << "GameRole";
break;
case QAudio::MusicRole:
dbg << "MusicRole";
break;
case QAudio::NotificationRole:
dbg << "NotificationRole";
break;
case QAudio::RingtoneRole:
dbg << "RingtoneRole";
break;
case QAudio::SonificationRole:
dbg << "SonificationRole";
break;
case QAudio::VideoRole:
dbg << "VideoRole";
break;
case QAudio::VoiceCommunicationRole:
dbg << "VoiceCommunicationRole";
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
|