2e67f95017
The rationale is explained in https://planet.kde.org/friedrich-kossebau-2023-06-28-include-also-moc-files-of-headers/ In case of KDEConnect, it impressively speeds up compilation. Before it took 390 seconds on a clean build and with this change it took 330 seconds. This is due to the mocs_compilation having to include the header files and thus all their headers. Due to the lots of small plugins we have, this means that the same headers must be compiled plenty of times. When we include the moc files directly in the C++ file, they are already available.
216 lines
8.3 KiB
C++
216 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 "moc_pausemusicplugin-win.cpp"
|
|
#include "pausemusicplugin-win.moc"
|