215 lines
8.3 KiB
C++
215 lines
8.3 KiB
C++
/**
|
|
* SPDX-FileCopyrightText: 2013 Albert Vaca <albertvaka@gmail.com>
|
|
* SPDX-FileCopyrightText: 2019 Piyush Aggarwal <piyushaggarwal002@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
|
*/
|
|
|
|
#include "pausemusicplugin-win.h"
|
|
|
|
#include "plugin_pausemusic_debug.h"
|
|
#include <KPluginFactory>
|
|
|
|
#include <Functiondiscoverykeys_devpkey.h>
|
|
|
|
K_PLUGIN_CLASS_WITH_JSON(PauseMusicPlugin, "kdeconnect_pausemusic.json")
|
|
|
|
PauseMusicPlugin::PauseMusicPlugin(QObject *parent, const QVariantList &args)
|
|
: KdeConnectPlugin(parent, args)
|
|
, sessionManager(GlobalSystemMediaTransportControlsSessionManager::RequestAsync().get())
|
|
{
|
|
CoInitialize(nullptr);
|
|
deviceEnumerator = nullptr;
|
|
HRESULT hr = CoCreateInstance(__uuidof(MMDeviceEnumerator), nullptr, CLSCTX_INPROC_SERVER, __uuidof(IMMDeviceEnumerator), (LPVOID *)&deviceEnumerator);
|
|
valid = (hr == S_OK);
|
|
if (!valid) {
|
|
qWarning("Initialization failed: Failed to create MMDeviceEnumerator");
|
|
qWarning("Error Code: %lx", hr);
|
|
}
|
|
}
|
|
|
|
bool PauseMusicPlugin::updateSinksList()
|
|
{
|
|
sinksList.clear();
|
|
if (!valid)
|
|
return false;
|
|
|
|
IMMDeviceCollection *devices = nullptr;
|
|
HRESULT hr = deviceEnumerator->EnumAudioEndpoints(eRender, DEVICE_STATE_ACTIVE, &devices);
|
|
|
|
if (hr != S_OK) {
|
|
qWarning("Failed to Enumumerate AudioEndpoints");
|
|
qWarning("Error Code: %lx", hr);
|
|
return false;
|
|
}
|
|
|
|
unsigned int deviceCount;
|
|
devices->GetCount(&deviceCount);
|
|
|
|
for (unsigned int i = 0; i < deviceCount; i++) {
|
|
IMMDevice *device = nullptr;
|
|
|
|
IPropertyStore *deviceProperties = nullptr;
|
|
PROPVARIANT deviceProperty;
|
|
QString name;
|
|
|
|
IAudioEndpointVolume *endpoint = nullptr;
|
|
|
|
// Get Properties
|
|
devices->Item(i, &device);
|
|
device->OpenPropertyStore(STGM_READ, &deviceProperties);
|
|
|
|
deviceProperties->GetValue(PKEY_Device_FriendlyName, &deviceProperty);
|
|
name = QString::fromWCharArray(deviceProperty.pwszVal);
|
|
// PropVariantClear(&deviceProperty);
|
|
|
|
hr = device->Activate(__uuidof(IAudioEndpointVolume), CLSCTX_ALL, NULL, (void **)&endpoint);
|
|
if (hr != S_OK) {
|
|
qWarning() << "Failed to create IAudioEndpointVolume for device:" << name;
|
|
qWarning("Error Code: %lx", hr);
|
|
|
|
device->Release();
|
|
continue;
|
|
}
|
|
|
|
// Register Callback
|
|
if (!sinksList.contains(name)) {
|
|
sinksList[name] = endpoint;
|
|
}
|
|
device->Release();
|
|
}
|
|
devices->Release();
|
|
return true;
|
|
}
|
|
|
|
void PauseMusicPlugin::updatePlayersList()
|
|
{
|
|
playersList.clear();
|
|
auto sessions = sessionManager.GetSessions();
|
|
for (uint32_t i = 0; i < sessions.Size(); i++) {
|
|
const auto player = sessions.GetAt(i);
|
|
auto playerName = player.SourceAppUserModelId();
|
|
|
|
QString uniqueName = QString::fromWCharArray(playerName.c_str());
|
|
for (int i = 2; playersList.contains(uniqueName); ++i) {
|
|
uniqueName += QStringLiteral(" [") + QString::number(i) + QStringLiteral("]");
|
|
}
|
|
playersList.insert(uniqueName, player);
|
|
}
|
|
}
|
|
|
|
PauseMusicPlugin::~PauseMusicPlugin()
|
|
{
|
|
CoUninitialize();
|
|
}
|
|
|
|
bool PauseMusicPlugin::receivePacket(const NetworkPacket &np)
|
|
{
|
|
bool pauseOnlyWhenTalking = config()->getBool(QStringLiteral("conditionTalking"), false);
|
|
|
|
if (pauseOnlyWhenTalking) {
|
|
if (np.get<QString>(QStringLiteral("event")) != QLatin1String("talking")) {
|
|
return true;
|
|
}
|
|
} else {
|
|
if (np.get<QString>(QStringLiteral("event")) != QLatin1String("ringing") && np.get<QString>(QStringLiteral("event")) != QLatin1String("talking")) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool pauseConditionFulfilled = !np.get<bool>(QStringLiteral("isCancel"));
|
|
|
|
bool pause = config()->getBool(QStringLiteral("actionPause"), true);
|
|
bool mute = config()->getBool(QStringLiteral("actionMute"), false);
|
|
|
|
const bool autoResume = config()->getBool(QStringLiteral("actionResume"), true);
|
|
|
|
if (pauseConditionFulfilled) {
|
|
if (mute) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Muting all the unmuted sinks";
|
|
this->updateSinksList();
|
|
QHashIterator<QString, IAudioEndpointVolume *> sinksIterator(sinksList);
|
|
while (sinksIterator.hasNext()) {
|
|
sinksIterator.next();
|
|
BOOL muted;
|
|
sinksIterator.value()->GetMute(&muted);
|
|
if (!((bool)muted)) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Trying to mute " << sinksIterator.key();
|
|
if (sinksIterator.value()->SetMute(true, NULL) == S_OK) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Muted " << sinksIterator.key();
|
|
mutedSinks.insert(sinksIterator.key());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pause) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Pausing all the playing media";
|
|
this->updatePlayersList();
|
|
QHashIterator<QString, GlobalSystemMediaTransportControlsSession> playersIterator(playersList);
|
|
while (playersIterator.hasNext()) {
|
|
playersIterator.next();
|
|
auto &player = playersIterator.value();
|
|
auto &playerName = playersIterator.key();
|
|
|
|
auto playbackInfo = player.GetPlaybackInfo();
|
|
auto playbackControls = playbackInfo.Controls();
|
|
if (playbackInfo.PlaybackStatus() == GlobalSystemMediaTransportControlsSessionPlaybackStatus::Playing) {
|
|
if (playbackControls.IsPauseEnabled()) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Trying to pause " << playerName;
|
|
if (player.TryPauseAsync().get()) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Paused " << playerName;
|
|
pausedSources.insert(playerName);
|
|
}
|
|
} else {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Pause not supported by the app! Trying to stop " << playerName;
|
|
if (player.TryStopAsync().get()) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Stopped " << playerName;
|
|
pausedSources.insert(playerName);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else if (autoResume) {
|
|
if (mute) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Unmuting sinks";
|
|
QHashIterator<QString, IAudioEndpointVolume *> sinksIterator(sinksList);
|
|
while (sinksIterator.hasNext()) {
|
|
sinksIterator.next();
|
|
if (mutedSinks.contains(sinksIterator.key())) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Trying to unmute " << sinksIterator.key();
|
|
if (sinksIterator.value()->SetMute(false, NULL) == S_OK) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Unmuted " << sinksIterator.key();
|
|
}
|
|
mutedSinks.remove(sinksIterator.key());
|
|
}
|
|
}
|
|
}
|
|
if (pause) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Unpausing media";
|
|
QHashIterator<QString, GlobalSystemMediaTransportControlsSession> playersIterator(playersList);
|
|
while (playersIterator.hasNext()) {
|
|
playersIterator.next();
|
|
auto &player = playersIterator.value();
|
|
auto &playerName = playersIterator.key();
|
|
|
|
auto playbackInfo = player.GetPlaybackInfo();
|
|
auto playbackControls = playbackInfo.Controls();
|
|
if (pausedSources.contains({playerName})) {
|
|
if (playbackInfo.PlaybackStatus() == GlobalSystemMediaTransportControlsSessionPlaybackStatus::Paused) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Trying to resume " << playerName;
|
|
if (player.TryPlayAsync().get()) {
|
|
qCDebug(KDECONNECT_PLUGIN_PAUSEMUSIC) << "Resumed " << playerName;
|
|
}
|
|
}
|
|
pausedSources.remove(playerName);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
#include "pausemusicplugin-win.moc"
|