/* This file is part of the KDE project Copyright (C) 2006-2008 Matthias Kretz This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) version 3, or any later version accepted by the membership of KDE e.V. (or its successor approved by the membership of KDE e.V.), Trolltech ASA (or its successors, if any) and the KDE Free Qt Foundation, which shall act as a proxy defined in Section 6 of version 3 of the license. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library. If not, see . */ #include "globalconfig_p.h" #include "factory_p.h" #include "objectdescription.h" #include "phonondefs_p.h" #include "platformplugin.h" #include "backendinterface.h" #include "qsettingsgroup_p.h" #include "phononnamespace_p.h" #include #include QT_BEGIN_NAMESPACE namespace Phonon { GlobalConfig::GlobalConfig() : m_config(QLatin1String("kde.org"), QLatin1String("libphonon")) { } GlobalConfig::~GlobalConfig() { } enum WhatToFilter { FilterAdvancedDevices = 1, FilterHardwareDevices = 2, FilterUnavailableDevices = 4 }; static void filter(ObjectDescriptionType type, BackendInterface *backendIface, QList *list, int whatToFilter) { QMutableListIterator it(*list); while (it.hasNext()) { const QHash properties = backendIface->objectDescriptionProperties(type, it.next()); QVariant var; if (whatToFilter & FilterAdvancedDevices) { var = properties.value("isAdvanced"); if (var.isValid() && var.toBool()) { it.remove(); continue; } } if (whatToFilter & FilterHardwareDevices) { var = properties.value("isHardwareDevice"); if (var.isValid() && var.toBool()) { it.remove(); continue; } } if (whatToFilter & FilterUnavailableDevices) { var = properties.value("available"); if (var.isValid() && !var.toBool()) { it.remove(); continue; } } } } static QList listSortedByConfig(const QSettingsGroup &backendConfig, Phonon::Category category, QList &defaultList) { if (defaultList.size() <= 1) { // nothing to sort return defaultList; } else { // make entries unique QSet seen; QMutableListIterator it(defaultList); while (it.hasNext()) { if (seen.contains(it.next())) { it.remove(); } else { seen.insert(it.value()); } } } QString categoryKey = QLatin1String("Category_") + QString::number(static_cast(category)); if (!backendConfig.hasKey(categoryKey)) { // no list in config for the given category categoryKey = QLatin1String("Category_") + QString::number(static_cast(Phonon::NoCategory)); if (!backendConfig.hasKey(categoryKey)) { // no list in config for NoCategory return defaultList; } } //Now the list from m_config QList deviceList = backendConfig.value(categoryKey, QList()); //if there are devices in m_config that the backend doesn't report, remove them from the list QMutableListIterator i(deviceList); while (i.hasNext()) { if (0 == defaultList.removeAll(i.next())) { i.remove(); } } //if the backend reports more devices that are not in m_config append them to the list deviceList += defaultList; return deviceList; } QList GlobalConfig::audioOutputDeviceListFor(Phonon::Category category, int override) const { //The devices need to be stored independently for every backend const QSettingsGroup backendConfig(&m_config, QLatin1String("AudioOutputDevice")); // + Factory::identifier()); const QSettingsGroup generalGroup(&m_config, QLatin1String("General")); const bool hideAdvancedDevices = ((override & AdvancedDevicesFromSettings) ? generalGroup.value(QLatin1String("HideAdvancedDevices"), true) : static_cast(override & HideAdvancedDevices)); QList defaultList; #ifndef QT_NO_PHONON_PLATFORMPLUGIN if (PlatformPlugin *platformPlugin = Factory::platformPlugin()) { // the platform plugin lists the audio devices for the platform // this list already is in default order (as defined by the platform plugin) defaultList = platformPlugin->objectDescriptionIndexes(Phonon::AudioOutputDeviceType); if (hideAdvancedDevices) { QMutableListIterator it(defaultList); while (it.hasNext()) { AudioOutputDevice objDesc = AudioOutputDevice::fromIndex(it.next()); const QVariant var = objDesc.property("isAdvanced"); if (var.isValid() && var.toBool()) { it.remove(); } } } } #endif //QT_NO_PHONON_PLATFORMPLUGIN // lookup the available devices directly from the backend (mostly for virtual devices) if (BackendInterface *backendIface = qobject_cast(Factory::backend())) { // this list already is in default order (as defined by the backend) QList list = backendIface->objectDescriptionIndexes(Phonon::AudioOutputDeviceType); if (hideAdvancedDevices || !defaultList.isEmpty() || (override & HideUnavailableDevices)) { filter(AudioOutputDeviceType, backendIface, &list, (hideAdvancedDevices ? FilterAdvancedDevices : 0) // the platform plugin already provided the hardware devices | (defaultList.isEmpty() ? 0 : FilterHardwareDevices) | ((override & HideUnavailableDevices) ? FilterUnavailableDevices : 0) ); } defaultList += list; } return listSortedByConfig(backendConfig, category, defaultList); } int GlobalConfig::audioOutputDeviceFor(Phonon::Category category, int override) const { QList ret = audioOutputDeviceListFor(category, override); if (ret.isEmpty()) return -1; return ret.first(); } #ifndef QT_NO_PHONON_AUDIOCAPTURE QList GlobalConfig::audioCaptureDeviceListFor(Phonon::Category category, int override) const { //The devices need to be stored independently for every backend const QSettingsGroup backendConfig(&m_config, QLatin1String("AudioCaptureDevice")); // + Factory::identifier()); const QSettingsGroup generalGroup(&m_config, QLatin1String("General")); const bool hideAdvancedDevices = ((override & AdvancedDevicesFromSettings) ? generalGroup.value(QLatin1String("HideAdvancedDevices"), true) : static_cast(override & HideAdvancedDevices)); QList defaultList; #ifndef QT_NO_PHONON_PLATFORMPLUGIN if (PlatformPlugin *platformPlugin = Factory::platformPlugin()) { // the platform plugin lists the audio devices for the platform // this list already is in default order (as defined by the platform plugin) defaultList = platformPlugin->objectDescriptionIndexes(Phonon::AudioCaptureDeviceType); if (hideAdvancedDevices) { QMutableListIterator it(defaultList); while (it.hasNext()) { AudioCaptureDevice objDesc = AudioCaptureDevice::fromIndex(it.next()); const QVariant var = objDesc.property("isAdvanced"); if (var.isValid() && var.toBool()) { it.remove(); } } } } #endif //QT_NO_PHONON_PLATFORMPLUGIN // lookup the available devices directly from the backend (mostly for virtual devices) if (BackendInterface *backendIface = qobject_cast(Factory::backend())) { // this list already is in default order (as defined by the backend) QList list = backendIface->objectDescriptionIndexes(Phonon::AudioCaptureDeviceType); if (hideAdvancedDevices || !defaultList.isEmpty() || (override & HideUnavailableDevices)) { filter(AudioCaptureDeviceType, backendIface, &list, (hideAdvancedDevices ? FilterAdvancedDevices : 0) // the platform plugin already provided the hardware devices | (defaultList.isEmpty() ? 0 : FilterHardwareDevices) | ((override & HideUnavailableDevices) ? FilterUnavailableDevices : 0) ); } defaultList += list; } return listSortedByConfig(backendConfig, category, defaultList); } int GlobalConfig::audioCaptureDeviceFor(Phonon::Category category, int override) const { QList ret = audioCaptureDeviceListFor(category, override); if (ret.isEmpty()) return -1; return ret.first(); } #endif //QT_NO_PHONON_AUDIOCAPTURE } // namespace Phonon QT_END_NAMESPACE