/*
* Copyright (C) 2015 Canonical, Ltd.
* Copyright (C) 2021 Robert Tari
*
* This program is free software: you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 3, as published
* by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranties of
* MERCHANTABILITY, SATISFACTORY QUALITY, or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program. If not, see .
*
* Authors:
* Xavi Garcia
* Robert Tari
*/
#include "dbus-pulse-volume.h"
#include "dbus_properties_interface.h"
#include "dbus_accounts_interface.h"
#include "dbus_accountssound_interface.h"
#include "stream_restore_interface.h"
#include
#include
unsigned int volumeDoubleToUint(double volume)
{
double tmp = (double)(PA_VOLUME_NORM - PA_VOLUME_MUTED) * volume;
return (unsigned int)tmp + PA_VOLUME_MUTED;
}
double volumeUIntToDoulbe(uint volume)
{
double tmp = (double)(volume - PA_VOLUME_MUTED);
return tmp / (double)(PA_VOLUME_NORM - PA_VOLUME_MUTED);
}
DBusPulseVolume::DBusPulseVolume() :
QObject()
{
std::unique_ptr basicConnectionInterface(new DBusPropertiesInterface("org.PulseAudio1",
"/org/pulseaudio/server_lookup1",
QDBusConnection::sessionBus(), 0));
QDBusReply connection_string = basicConnectionInterface->call(QLatin1String("Get"),
QLatin1String("org.PulseAudio.ServerLookup1"),
QLatin1String("Address"));
if (!connection_string.isValid())
{
qWarning() << "DBusPulseVolume::DBusPulseVolume(): D-Bus error: " << connection_string.error().message();
}
connection_.reset(new QDBusConnection(QDBusConnection::connectToPeer(connection_string.value().toString(), "set-volume")));
if (connection_->isConnected())
{
interface_paths_.reset(new StreamRestoreInterface("org.PulseAudio.Ext.StreamRestore1",
"/org/pulseaudio/stream_restore1",
*(connection_.get()), 0));
if (interface_paths_)
{
// get the role paths
fillRolePath("multimedia");
fillRolePath("alert");
fillRolePath("alarm");
fillRolePath("phone");
}
initializeAccountsInterface();
}
else
{
qWarning() << "DBusPulseVolume::DBusPulseVolume(): Error connecting: " << connection_->lastError().message();
}
}
DBusPulseVolume::~DBusPulseVolume()
{
}
QString DBusPulseVolume::fillRolePath(QString const &role)
{
QString role_complete_name = QString("sink-input-by-media-role:") + role;
// get the role paths
QDBusReply objectPath = interface_paths_->call(QLatin1String("GetEntryByName"), role_complete_name);
if (!objectPath.isValid())
{
qWarning() << "SetVolume::fillRolePath(): D-Bus error: " << objectPath.error().message();
return "";
}
auto role_info = std::make_shared();
role_info->interface_.reset(new DBusPropertiesInterface("org.PulseAudio.Ext.StreamRestore1.RestoreEntry",
objectPath.value().path(),
*(connection_.get()), 0));
if (!role_info->interface_)
{
qWarning() << "SetVolume::fillRolePath() - Error obtaining properties interface";
return "";
}
role_info->path_ = objectPath.value().path();
roles_map_[role] = role_info;
return role_info->path_;
}
bool DBusPulseVolume::setVolume(QString const & role, double volume)
{
if (!interface_paths_)
{
qWarning() << "SetVolume::setVolume(): error: Volume interfaces are not initialized";
return false;
}
RolesMap::const_iterator iter = roles_map_.find(role);
if (iter != roles_map_.end())
{
QVariant var;
PulseaudioVolumeArray t;
PulseaudioVolume vv(0, volumeDoubleToUint(volume));
t.addItem(vv);
var.setValue(t);
QDBusVariant dbusVar(var);
QDBusReply set_vol = (*iter).second->interface_->call(QLatin1String("Set"),
QVariant::fromValue(QString("org.PulseAudio.Ext.StreamRestore1.RestoreEntry")),
QVariant::fromValue(QString("Volume")),
QVariant::fromValue(dbusVar));
if (!set_vol.isValid())
{
qWarning() << "SetVolume::setVolume(): D-Bus error: " << set_vol.error().message();
return false;
}
if (accounts_interface_)
{
QDBusVariant dbusVar(QVariant::fromValue(volume));
QDBusReply set_vol = accounts_interface_->call(QLatin1String("Set"),
QVariant::fromValue(QString("com.lomiri.AccountsService.Sound")),
QVariant::fromValue(QString("Volume")),
QVariant::fromValue(dbusVar));
if (!set_vol.isValid())
{
qWarning() << "SetVolume::setVolume(): D-Bus error: " << set_vol.error().message();
return false;
}
}
}
else
{
qWarning() << "SetVolume::setVolume(): role " << role << " was not found.";
return false;
}
return true;
}
double DBusPulseVolume::getVolume(QString const & role)
{
if (interface_paths_)
{
RolesMap::const_iterator iter = roles_map_.find(role);
if (iter != roles_map_.end())
{
QDBusReply prev_vol = (*iter).second->interface_->call(QLatin1String("Get"),
QLatin1String("org.PulseAudio.Ext.StreamRestore1.RestoreEntry"),
QLatin1String("Volume"));
if (!prev_vol.isValid())
{
qWarning() << "SetVolume::setVolume(): D-Bus error: " << prev_vol.error().message();
}
QDBusArgument arg = prev_vol.value().value();
PulseaudioVolumeArray element;
arg >> element;
return volumeUIntToDoulbe(element.getItem(0).getVolume());
}
}
return -1.0;
}
void DBusPulseVolume::initializeAccountsInterface()
{
auto username = qgetenv("USER");
if (username != "")
{
qDebug() << "Setting Accounts interface for user: " << username;
std::unique_ptr setInterface(new AccountsInterface("org.freedesktop.Accounts",
"/org/freedesktop/Accounts",
QDBusConnection::systemBus(), 0));
QDBusReply userResp = setInterface->call(QLatin1String("FindUserByName"),
QLatin1String(username));
if (!userResp.isValid())
{
qWarning() << "SetVolume::initializeAccountsInterface(): D-Bus error: " << userResp.error().message();
}
auto userPath = userResp.value().path();
if (userPath != "")
{
std::unique_ptr soundInterface(new AccountsSoundInterface("org.freedesktop.Accounts",
userPath,
QDBusConnection::systemBus(), 0));
accounts_interface_.reset(new DBusPropertiesInterface("org.freedesktop.Accounts",
userPath,
soundInterface->connection(), 0));
if (!accounts_interface_->isValid())
{
qWarning() << "SetVolume::initializeAccountsInterface(): D-Bus error: " << accounts_interface_->lastError().message();
}
signal_spy_volume_changed_.reset(new QSignalSpy(accounts_interface_.get(),&DBusPropertiesInterface::PropertiesChanged));
}
}
}
bool DBusPulseVolume::waitForVolumeChangedInAccountsService()
{
if (signal_spy_volume_changed_)
{
return signal_spy_volume_changed_->wait();
}
else
{
qWarning() << "DBusPulseVolume::waitForVolumeChangedInAccountsService(): signal was not instantiated";
}
return false;
}