// Copyright (C) 2021 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause #include "audiosource.h" #include #include #include #include #include #include #include #include #include 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(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)) { initializeWindow(); initializeAudio(QMediaDevices::defaultAudioInput()); } 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::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_audioInfo->stop(); m_audioInput->stop(); m_audioInput->disconnect(this); initializeAudio(m_deviceBox->itemData(index).value()); } void InputTest::sliderChanged(int value) { qreal linearVolume = QAudio::convertVolume(value / qreal(100), QAudio::LogarithmicVolumeScale, QAudio::LinearVolumeScale); m_audioInput->setVolume(linearVolume); }