summaryrefslogtreecommitdiffstats
path: root/examples/multimedia/audiosource/audiosource.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'examples/multimedia/audiosource/audiosource.cpp')
-rw-r--r--examples/multimedia/audiosource/audiosource.cpp251
1 files changed, 251 insertions, 0 deletions
diff --git a/examples/multimedia/audiosource/audiosource.cpp b/examples/multimedia/audiosource/audiosource.cpp
new file mode 100644
index 000000000..48e33fa2d
--- /dev/null
+++ b/examples/multimedia/audiosource/audiosource.cpp
@@ -0,0 +1,251 @@
+// Copyright (C) 2021 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+
+#include "audiosource.h"
+
+#include <QAudioDevice>
+#include <QAudioSource>
+#include <QDateTime>
+#include <QDebug>
+#include <QLabel>
+#include <QPainter>
+#include <QVBoxLayout>
+#include <QtEndian>
+
+#if QT_CONFIG(permissions)
+ #include <QCoreApplication>
+ #include <QPermission>
+#endif
+
+#include <math.h>
+#include <stdlib.h>
+
+AudioInfo::AudioInfo(const QAudioFormat &format) : m_format(format) { }
+
+void AudioInfo::start()
+{
+ open(QIODevice::WriteOnly);
+}
+
+void AudioInfo::stop()
+{
+ close();
+}
+
+qint64 AudioInfo::readData(char * /* data */, qint64 /* maxlen */)
+{
+ return 0;
+}
+
+qreal AudioInfo::calculateLevel(const char *data, qint64 len) const
+{
+ const int channelBytes = m_format.bytesPerSample();
+ const int sampleBytes = m_format.bytesPerFrame();
+ const int numSamples = len / sampleBytes;
+
+ float maxValue = 0;
+ auto *ptr = reinterpret_cast<const unsigned char *>(data);
+
+ for (int i = 0; i < numSamples; ++i) {
+ for (int j = 0; j < m_format.channelCount(); ++j) {
+ float value = m_format.normalizedSampleValue(ptr);
+
+ maxValue = qMax(value, maxValue);
+ ptr += channelBytes;
+ }
+ }
+ return maxValue;
+}
+
+qint64 AudioInfo::writeData(const char *data, qint64 len)
+{
+ m_level = calculateLevel(data, len);
+
+ emit levelChanged(m_level);
+
+ return len;
+}
+
+RenderArea::RenderArea(QWidget *parent) : QWidget(parent)
+{
+ setBackgroundRole(QPalette::Base);
+ setAutoFillBackground(true);
+
+ setMinimumHeight(30);
+ setMinimumWidth(200);
+}
+
+void RenderArea::paintEvent(QPaintEvent * /* event */)
+{
+ QPainter painter(this);
+
+ painter.setPen(Qt::black);
+
+ const QRect frame = painter.viewport() - QMargins(10, 10, 10, 10);
+ painter.drawRect(frame);
+ if (m_level == 0.0)
+ return;
+
+ const int pos = qRound(qreal(frame.width() - 1) * m_level);
+ painter.fillRect(frame.left() + 1, frame.top() + 1, pos, frame.height() - 1, Qt::red);
+}
+
+void RenderArea::setLevel(qreal value)
+{
+ m_level = value;
+ update();
+}
+
+InputTest::InputTest() : m_devices(new QMediaDevices(this))
+{
+ init();
+}
+
+void InputTest::initializeWindow()
+{
+ QVBoxLayout *layout = new QVBoxLayout(this);
+
+ m_canvas = new RenderArea(this);
+ layout->addWidget(m_canvas);
+
+ m_deviceBox = new QComboBox(this);
+ const QAudioDevice &defaultDeviceInfo = QMediaDevices::defaultAudioInput();
+ m_deviceBox->addItem(defaultDeviceInfo.description(), QVariant::fromValue(defaultDeviceInfo));
+ for (auto &deviceInfo : m_devices->audioInputs()) {
+ if (deviceInfo != defaultDeviceInfo)
+ m_deviceBox->addItem(deviceInfo.description(), QVariant::fromValue(deviceInfo));
+ }
+
+ connect(m_deviceBox, &QComboBox::activated, this, &InputTest::deviceChanged);
+ layout->addWidget(m_deviceBox);
+
+ m_volumeSlider = new QSlider(Qt::Horizontal, this);
+ m_volumeSlider->setRange(0, 100);
+ m_volumeSlider->setValue(100);
+ connect(m_volumeSlider, &QSlider::valueChanged, this, &InputTest::sliderChanged);
+ layout->addWidget(m_volumeSlider);
+
+ m_modeButton = new QPushButton(this);
+ connect(m_modeButton, &QPushButton::clicked, this, &InputTest::toggleMode);
+ layout->addWidget(m_modeButton);
+
+ m_suspendResumeButton = new QPushButton(this);
+ connect(m_suspendResumeButton, &QPushButton::clicked, this, &InputTest::toggleSuspend);
+ layout->addWidget(m_suspendResumeButton);
+}
+
+void InputTest::initializeAudio(const QAudioDevice &deviceInfo)
+{
+ QAudioFormat format;
+ format.setSampleRate(8000);
+ format.setChannelCount(1);
+ format.setSampleFormat(QAudioFormat::Int16);
+
+ m_audioInfo.reset(new AudioInfo(format));
+ connect(m_audioInfo.data(), &AudioInfo::levelChanged, m_canvas, &RenderArea::setLevel);
+
+ m_audioInput.reset(new QAudioSource(deviceInfo, format));
+ qreal initialVolume = QAudio::convertVolume(m_audioInput->volume(), QAudio::LinearVolumeScale,
+ QAudio::LogarithmicVolumeScale);
+ m_volumeSlider->setValue(qRound(initialVolume * 100));
+ m_audioInfo->start();
+ toggleMode();
+}
+
+void InputTest::initializeErrorWindow()
+{
+ QVBoxLayout *layout = new QVBoxLayout(this);
+ QLabel *errorLabel = new QLabel(tr("Microphone permission is not granted!"));
+ errorLabel->setWordWrap(true);
+ errorLabel->setAlignment(Qt::AlignCenter);
+ layout->addWidget(errorLabel);
+}
+
+void InputTest::init()
+{
+#if QT_CONFIG(permissions)
+ QMicrophonePermission microphonePermission;
+ switch (qApp->checkPermission(microphonePermission)) {
+ case Qt::PermissionStatus::Undetermined:
+ qApp->requestPermission(microphonePermission, this, &InputTest::init);
+ return;
+ case Qt::PermissionStatus::Denied:
+ qWarning("Microphone permission is not granted!");
+ initializeErrorWindow();
+ return;
+ case Qt::PermissionStatus::Granted:
+ break;
+ }
+#endif
+ initializeWindow();
+ initializeAudio(QMediaDevices::defaultAudioInput());
+}
+
+void InputTest::toggleMode()
+{
+ m_audioInput->stop();
+ toggleSuspend();
+
+ // Change between pull and push modes
+ if (m_pullMode) {
+ m_modeButton->setText(tr("Enable push mode"));
+ m_audioInput->start(m_audioInfo.data());
+ } else {
+ m_modeButton->setText(tr("Enable pull mode"));
+ auto *io = m_audioInput->start();
+ if (!io)
+ return;
+
+ connect(io, &QIODevice::readyRead, [this, io]() {
+ static const qint64 BufferSize = 4096;
+ const qint64 len = qMin(m_audioInput->bytesAvailable(), BufferSize);
+
+ QByteArray buffer(len, 0);
+ qint64 l = io->read(buffer.data(), len);
+ if (l > 0) {
+ const qreal level = m_audioInfo->calculateLevel(buffer.constData(), l);
+ m_canvas->setLevel(level);
+ }
+ });
+ }
+
+ m_pullMode = !m_pullMode;
+}
+
+void InputTest::toggleSuspend()
+{
+ // toggle suspend/resume
+ switch (m_audioInput->state()) {
+ case QAudio::SuspendedState:
+ case QAudio::StoppedState:
+ m_audioInput->resume();
+ m_suspendResumeButton->setText(tr("Suspend recording"));
+ break;
+ case QAudio::ActiveState:
+ m_audioInput->suspend();
+ m_suspendResumeButton->setText(tr("Resume recording"));
+ break;
+ case QAudio::IdleState:
+ // no-op
+ break;
+ }
+}
+
+void InputTest::deviceChanged(int index)
+{
+ m_audioInput->stop();
+ m_audioInput->disconnect(this);
+ m_audioInfo->stop();
+
+ initializeAudio(m_deviceBox->itemData(index).value<QAudioDevice>());
+}
+
+void InputTest::sliderChanged(int value)
+{
+ qreal linearVolume = QAudio::convertVolume(value / qreal(100), QAudio::LogarithmicVolumeScale,
+ QAudio::LinearVolumeScale);
+
+ m_audioInput->setVolume(linearVolume);
+}
+
+#include "moc_audiosource.cpp"