2019-02-03 00:50:36 +00:00
|
|
|
/**
|
2020-08-17 10:48:10 +01:00
|
|
|
* SPDX-FileCopyrightText: 2018 Jun Bo Bi <jambonmcyeah@gmail.com>
|
2019-02-03 00:50:36 +00:00
|
|
|
*
|
2020-08-17 10:48:10 +01:00
|
|
|
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
2019-02-03 00:50:36 +00:00
|
|
|
*/
|
2018-11-20 00:14:17 +00:00
|
|
|
|
2019-02-03 00:50:36 +00:00
|
|
|
#include "systemvolumeplugin-win.h"
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QJsonArray>
|
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonObject>
|
|
|
|
|
2019-02-03 00:50:36 +00:00
|
|
|
#include <KPluginFactory>
|
|
|
|
|
2018-11-20 00:14:17 +00:00
|
|
|
#include <Functiondiscoverykeys_devpkey.h>
|
|
|
|
|
2019-02-03 00:50:36 +00:00
|
|
|
#include <core/device.h>
|
|
|
|
|
2020-05-26 17:55:47 +01:00
|
|
|
#include "plugin_systemvolume_debug.h"
|
2018-11-20 00:14:17 +00:00
|
|
|
|
2020-05-26 17:55:47 +01:00
|
|
|
K_PLUGIN_CLASS_WITH_JSON(SystemvolumePlugin, "kdeconnect_systemvolume.json")
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
// Private classes of SystemvolumePlugin
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
class SystemvolumePlugin::CAudioEndpointVolumeCallback : public IAudioEndpointVolumeCallback
|
2018-11-20 00:14:17 +00:00
|
|
|
{
|
2020-11-19 19:54:25 +00:00
|
|
|
LONG _cRef;
|
2018-11-20 00:14:17 +00:00
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
public:
|
|
|
|
CAudioEndpointVolumeCallback(SystemvolumePlugin &x, QString sinkName) : enclosing(x), name(std::move(sinkName)), _cRef(1) {}
|
|
|
|
~CAudioEndpointVolumeCallback(){};
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
// IUnknown methods -- AddRef, Release, and QueryInterface
|
|
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE AddRef() override
|
|
|
|
{
|
|
|
|
return InterlockedIncrement(&_cRef);
|
|
|
|
}
|
|
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE Release() override
|
|
|
|
{
|
|
|
|
ULONG ulRef = InterlockedDecrement(&_cRef);
|
|
|
|
if (ulRef == 0)
|
|
|
|
{
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
return ulRef;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, VOID **ppvInterface) override
|
|
|
|
{
|
|
|
|
if (IID_IUnknown == riid)
|
|
|
|
{
|
|
|
|
AddRef();
|
|
|
|
*ppvInterface = (IUnknown *)this;
|
|
|
|
}
|
|
|
|
else if (__uuidof(IMMNotificationClient) == riid)
|
|
|
|
{
|
|
|
|
AddRef();
|
|
|
|
*ppvInterface = (IMMNotificationClient *)this;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*ppvInterface = NULL;
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
// Callback method for endpoint-volume-change notifications.
|
2018-11-20 00:14:17 +00:00
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
HRESULT STDMETHODCALLTYPE OnNotify(PAUDIO_VOLUME_NOTIFICATION_DATA pNotify) override
|
2018-11-20 00:14:17 +00:00
|
|
|
{
|
2020-11-19 19:54:25 +00:00
|
|
|
NetworkPacket np(PACKET_TYPE_SYSTEMVOLUME);
|
|
|
|
np.set<int>(QStringLiteral("volume"), (int)(pNotify->fMasterVolume * 100));
|
|
|
|
np.set<bool>(QStringLiteral("muted"), pNotify->bMuted);
|
|
|
|
np.set<QString>(QStringLiteral("name"), name);
|
|
|
|
enclosing.sendPacket(np);
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
private:
|
2018-11-20 00:14:17 +00:00
|
|
|
SystemvolumePlugin &enclosing;
|
2020-11-19 19:54:25 +00:00
|
|
|
QString name;
|
2018-11-20 00:14:17 +00:00
|
|
|
};
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
class SystemvolumePlugin::CMMNotificationClient : public IMMNotificationClient
|
2018-11-20 00:14:17 +00:00
|
|
|
{
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
public:
|
|
|
|
CMMNotificationClient(SystemvolumePlugin &x) : enclosing(x), _cRef(1){};
|
|
|
|
|
|
|
|
~CMMNotificationClient(){};
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
// IUnknown methods -- AddRef, Release, and QueryInterface
|
|
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE AddRef() override
|
|
|
|
{
|
|
|
|
return InterlockedIncrement(&_cRef);
|
|
|
|
}
|
|
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE Release() override
|
|
|
|
{
|
|
|
|
ULONG ulRef = InterlockedDecrement(&_cRef);
|
|
|
|
if (ulRef == 0)
|
|
|
|
{
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
return ulRef;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, VOID **ppvInterface) override
|
|
|
|
{
|
|
|
|
if (IID_IUnknown == riid)
|
|
|
|
{
|
|
|
|
AddRef();
|
|
|
|
*ppvInterface = (IUnknown *)this;
|
|
|
|
}
|
|
|
|
else if (__uuidof(IMMNotificationClient) == riid)
|
|
|
|
{
|
|
|
|
AddRef();
|
|
|
|
*ppvInterface = (IMMNotificationClient *)this;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*ppvInterface = NULL;
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
// Callback methods for device-event notifications.
|
2018-11-20 00:14:17 +00:00
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
HRESULT STDMETHODCALLTYPE OnDefaultDeviceChanged(EDataFlow flow, ERole role, LPCWSTR pwstrDeviceId) override
|
2018-11-20 00:14:17 +00:00
|
|
|
{
|
2020-11-19 19:54:25 +00:00
|
|
|
if (flow == eRender)
|
|
|
|
{
|
|
|
|
enclosing.sendSinkList();
|
|
|
|
}
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE OnDeviceAdded(LPCWSTR pwstrDeviceId) override
|
|
|
|
{
|
|
|
|
enclosing.sendSinkList();
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct RemovedDeviceThreadData {
|
|
|
|
|
|
|
|
QString qDeviceId;
|
|
|
|
SystemvolumePlugin* plugin;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
static DWORD WINAPI releaseRemovedDevice(_In_ LPVOID lpParameter) {
|
|
|
|
|
|
|
|
auto* data = static_cast<RemovedDeviceThreadData*>(lpParameter);
|
|
|
|
|
|
|
|
if (!data->plugin->sinkList.empty())
|
|
|
|
{
|
|
|
|
auto idToNameIterator = data->plugin->idToNameMap.find(data->qDeviceId);
|
|
|
|
if (idToNameIterator == data->plugin->idToNameMap.end()) return 0;
|
|
|
|
|
|
|
|
QString& sinkName = idToNameIterator.value();
|
|
|
|
|
|
|
|
auto sinkListIterator = data->plugin->sinkList.find(sinkName);
|
|
|
|
if (sinkListIterator == data->plugin->sinkList.end()) return 0;
|
|
|
|
|
|
|
|
auto& sink = sinkListIterator.value();
|
|
|
|
|
|
|
|
sink.first->UnregisterControlChangeNotify(sink.second);
|
|
|
|
sink.first->Release();
|
|
|
|
sink.second->Release();
|
|
|
|
|
|
|
|
data->plugin->idToNameMap.remove(data->qDeviceId);
|
|
|
|
data->plugin->sinkList.remove(sinkName);
|
|
|
|
}
|
|
|
|
|
|
|
|
data->plugin->sendSinkList();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE OnDeviceRemoved(LPCWSTR pwstrDeviceId) override
|
|
|
|
{
|
|
|
|
static RemovedDeviceThreadData data {};
|
|
|
|
data.qDeviceId = QString::fromWCharArray(pwstrDeviceId);
|
|
|
|
data.plugin = &enclosing;
|
|
|
|
|
|
|
|
DWORD threadId;
|
|
|
|
HANDLE threadHandle = CreateThread(NULL, 0, releaseRemovedDevice, &data, 0, &threadId);
|
|
|
|
CloseHandle(threadHandle);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE OnDeviceStateChanged(LPCWSTR pwstrDeviceId, DWORD dwNewState) override
|
|
|
|
{
|
|
|
|
if (dwNewState == DEVICE_STATE_UNPLUGGED) return OnDeviceRemoved(pwstrDeviceId);
|
|
|
|
|
|
|
|
enclosing.sendSinkList();
|
|
|
|
return S_OK;
|
|
|
|
}
|
2018-11-20 00:14:17 +00:00
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
HRESULT STDMETHODCALLTYPE OnPropertyValueChanged(LPCWSTR pwstrDeviceId, const PROPERTYKEY key) override
|
|
|
|
{
|
|
|
|
enclosing.sendSinkList();
|
2018-11-20 00:14:17 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2020-11-19 19:54:25 +00:00
|
|
|
private:
|
|
|
|
LONG _cRef;
|
2018-11-20 00:14:17 +00:00
|
|
|
SystemvolumePlugin &enclosing;
|
|
|
|
};
|
|
|
|
|
|
|
|
SystemvolumePlugin::SystemvolumePlugin(QObject *parent, const QVariantList &args)
|
|
|
|
: KdeConnectPlugin(parent, args),
|
|
|
|
sinkList()
|
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SystemvolumePlugin::~SystemvolumePlugin()
|
|
|
|
{
|
|
|
|
if (valid)
|
|
|
|
{
|
|
|
|
deviceEnumerator->UnregisterEndpointNotificationCallback(deviceCallback);
|
|
|
|
deviceEnumerator->Release();
|
|
|
|
deviceEnumerator = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SystemvolumePlugin::sendSinkList()
|
|
|
|
{
|
|
|
|
if (!valid)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
QJsonDocument document;
|
|
|
|
QJsonArray array;
|
|
|
|
|
|
|
|
IMMDeviceCollection *devices = nullptr;
|
2020-11-19 19:54:25 +00:00
|
|
|
HRESULT hr = deviceEnumerator->EnumAudioEndpoints(eRender, DEVICE_STATE_ACTIVE, &devices);
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
QString desc;
|
2020-11-19 19:54:25 +00:00
|
|
|
LPWSTR deviceId;
|
2018-11-20 00:14:17 +00:00
|
|
|
float volume;
|
|
|
|
BOOL muted;
|
|
|
|
|
|
|
|
IAudioEndpointVolume *endpoint = nullptr;
|
|
|
|
CAudioEndpointVolumeCallback *callback;
|
|
|
|
|
|
|
|
// Get Properties
|
|
|
|
devices->Item(i, &device);
|
|
|
|
device->OpenPropertyStore(STGM_READ, &deviceProperties);
|
|
|
|
|
|
|
|
deviceProperties->GetValue(PKEY_Device_FriendlyName, &deviceProperty);
|
|
|
|
name = QString::fromWCharArray(deviceProperty.pwszVal);
|
|
|
|
//PropVariantClear(&deviceProperty);
|
|
|
|
|
2019-04-30 18:09:26 +01:00
|
|
|
#ifndef __MINGW32__
|
2018-11-20 00:14:17 +00:00
|
|
|
deviceProperties->GetValue(PKEY_Device_DeviceDesc, &deviceProperty);
|
|
|
|
desc = QString::fromWCharArray(deviceProperty.pwszVal);
|
|
|
|
//PropVariantClear(&deviceProperty);
|
2019-04-30 18:09:26 +01:00
|
|
|
#endif
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
QJsonObject sinkObject;
|
2019-06-10 15:40:28 +01:00
|
|
|
sinkObject.insert(QStringLiteral("name"), name);
|
|
|
|
sinkObject.insert(QStringLiteral("description"), desc);
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2020-11-19 19:54:25 +00:00
|
|
|
|
|
|
|
device->GetId(&deviceId);
|
2018-11-20 00:14:17 +00:00
|
|
|
endpoint->GetMasterVolumeLevelScalar(&volume);
|
|
|
|
endpoint->GetMute(&muted);
|
|
|
|
|
2019-06-10 15:40:28 +01:00
|
|
|
sinkObject.insert(QStringLiteral("muted"), (bool)muted);
|
|
|
|
sinkObject.insert(QStringLiteral("volume"), (qint64)(volume * 100));
|
|
|
|
sinkObject.insert(QStringLiteral("maxVolume"), (qint64)100);
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
// Register Callback
|
2020-11-19 19:54:25 +00:00
|
|
|
QString qDeviceId = QString::fromWCharArray(deviceId);
|
|
|
|
if (!sinkList.contains(qDeviceId)) {
|
|
|
|
callback = new CAudioEndpointVolumeCallback(*this, name);
|
|
|
|
endpoint->RegisterControlChangeNotify(callback);
|
|
|
|
sinkList[name] = qMakePair(endpoint, callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
idToNameMap[qDeviceId] = name;
|
2018-11-20 00:14:17 +00:00
|
|
|
|
|
|
|
device->Release();
|
|
|
|
array.append(sinkObject);
|
|
|
|
}
|
|
|
|
devices->Release();
|
|
|
|
|
|
|
|
document.setArray(array);
|
|
|
|
|
|
|
|
NetworkPacket np(PACKET_TYPE_SYSTEMVOLUME);
|
|
|
|
np.set<QJsonDocument>(QStringLiteral("sinkList"), document);
|
|
|
|
sendPacket(np);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SystemvolumePlugin::connected()
|
|
|
|
{
|
|
|
|
if (!valid)
|
|
|
|
return;
|
|
|
|
|
|
|
|
deviceCallback = new CMMNotificationClient(*this);
|
|
|
|
deviceEnumerator->RegisterEndpointNotificationCallback(deviceCallback);
|
|
|
|
sendSinkList();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SystemvolumePlugin::receivePacket(const NetworkPacket &np)
|
|
|
|
{
|
|
|
|
if (!valid)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (np.has(QStringLiteral("requestSinks")))
|
|
|
|
{
|
|
|
|
return sendSinkList();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QString name = np.get<QString>(QStringLiteral("name"));
|
|
|
|
|
|
|
|
if (sinkList.contains(name))
|
|
|
|
{
|
|
|
|
if (np.has(QStringLiteral("volume")))
|
|
|
|
{
|
|
|
|
sinkList[name].first->SetMasterVolumeLevelScalar((float)np.get<int>(QStringLiteral("volume")) / 100, NULL);
|
|
|
|
}
|
|
|
|
if (np.has(QStringLiteral("muted")))
|
|
|
|
{
|
|
|
|
sinkList[name].first->SetMute(np.get<bool>(QStringLiteral("muted")), NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-02-03 00:50:36 +00:00
|
|
|
#include "systemvolumeplugin-win.moc"
|