kde-extraapps/plasma-nm/libs/models/networkmodel.cpp
2015-04-07 21:43:03 +00:00

1129 lines
48 KiB
C++

/*
Copyright 2013-2014 Jan Grulich <jgrulich@redhat.com>
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.), 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 <http://www.gnu.org/licenses/>.
*/
#include "networkmodel.h"
#include "networkmodelitem.h"
#include "debug.h"
#include "uiutils.h"
#if WITH_MODEMMANAGER_SUPPORT
#include <ModemManagerQt/manager.h>
#include <ModemManagerQt/modem.h>
#endif
#include <NetworkManagerQt/Settings>
#include <NetworkManagerQt/WimaxSetting>
NetworkModel::NetworkModel(QObject* parent)
: QAbstractListModel(parent)
{
QHash<int, QByteArray> roles = roleNames();
roles[ConnectionDetailsRole] = "ConnectionDetails";
roles[ConnectionIconRole] = "ConnectionIcon";
roles[ConnectionPathRole] = "ConnectionPath";
roles[ConnectionStateRole] = "ConnectionState";
roles[DeviceNameRole] = "DeviceName";
roles[DevicePathRole] = "DevicePath";
roles[DeviceStateRole] = "DeviceState";
roles[DuplicateRole] = "Duplicate";
roles[ItemUniqueNameRole] = "ItemUniqueName";
roles[ItemTypeRole] = "ItemType";
roles[LastUsedRole] = "LastUsed";
roles[LastUsedDateOnlyRole] = "LastUsedDateOnly";
roles[NameRole] = "Name";
roles[NspRole] = "Nsp";
roles[SectionRole] = "Section";
roles[SignalRole] = "Signal";
roles[SlaveRole] = "Slave";
roles[SsidRole] = "Ssid";
roles[SpecificPathRole] = "SpecificPath";
roles[SecurityTypeRole] = "SecurityType";
roles[SecurityTypeStringRole] = "SecurityTypeString";
roles[TimeStampRole] = "TimeStamp";
roles[TypeRole] = "Type";
roles[UniRole] = "Uni";
roles[UuidRole] = "Uuid";
roles[VpnState] = "VpnState";
setRoleNames(roles);
initialize();
}
NetworkModel::~NetworkModel()
{
}
QVariant NetworkModel::data(const QModelIndex& index, int role) const
{
const int row = index.row();
if (row >= 0 && row < m_list.count()) {
NetworkModelItem * item = m_list.itemAt(row);
switch (role) {
case ConnectionDetailsRole:
return item->details();
case ConnectionIconRole:
return item->icon();
case ConnectionPathRole:
return item->connectionPath();
case ConnectionStateRole:
return item->connectionState();
case DeviceNameRole:
return item->deviceName();
case DevicePathRole:
return item->devicePath();
case DeviceStateRole:
return item->deviceState();
case DuplicateRole:
return item->duplicate();
case ItemUniqueNameRole:
if (m_list.returnItems(NetworkItemsList::Name, item->name()).count() > 1) {
return item->originalName();
} else {
return item->name();
}
case ItemTypeRole:
return item->itemType();
case LastUsedRole:
return UiUtils::formatLastUsedDateRelative(item->timestamp());
case LastUsedDateOnlyRole:
return UiUtils::formatDateRelative(item->timestamp());
case NameRole:
return item->name();
case NspRole:
return item->nsp();
case SectionRole:
return item->sectionType();
case SignalRole:
return item->signal();
case SlaveRole:
return item->slave();
case SsidRole:
return item->ssid();
case SpecificPathRole:
return item->specificPath();
case SecurityTypeRole:
return item->securityType();
case SecurityTypeStringRole:
return UiUtils::labelFromWirelessSecurity(item->securityType());
case TimeStampRole:
return item->timestamp();
case TypeRole:
return item->type();
case UniRole:
return item->uni();
case UuidRole:
return item->uuid();
case VpnState:
return item->vpnState();
default:
break;
}
}
return QVariant();
}
int NetworkModel::rowCount(const QModelIndex& parent) const
{
Q_UNUSED(parent);
return m_list.count();
}
void NetworkModel::initialize()
{
// Initialize existing connections
foreach (const NetworkManager::Connection::Ptr& connection, NetworkManager::listConnections()) {
addConnection(connection);
}
// Initialize existing devices
foreach (const NetworkManager::Device::Ptr& dev, NetworkManager::networkInterfaces()) {
addDevice(dev);
}
// Initialize existing active connections
foreach (const NetworkManager::ActiveConnection::Ptr& active, NetworkManager::activeConnections()) {
addActiveConnection(active);
}
initializeSignals();
}
void NetworkModel::initializeSignals()
{
connect(NetworkManager::notifier(), SIGNAL(activeConnectionAdded(QString)),
SLOT(activeConnectionAdded(QString)), Qt::UniqueConnection);
connect(NetworkManager::notifier(), SIGNAL(activeConnectionRemoved(QString)),
SLOT(activeConnectionRemoved(QString)), Qt::UniqueConnection);
connect(NetworkManager::settingsNotifier(), SIGNAL(connectionAdded(QString)),
SLOT(connectionAdded(QString)), Qt::UniqueConnection);
connect(NetworkManager::settingsNotifier(), SIGNAL(connectionRemoved(QString)),
SLOT(connectionRemoved(QString)), Qt::UniqueConnection);
connect(NetworkManager::notifier(), SIGNAL(deviceAdded(QString)),
SLOT(deviceAdded(QString)), Qt::UniqueConnection);
connect(NetworkManager::notifier(), SIGNAL(deviceRemoved(QString)),
SLOT(deviceRemoved(QString)), Qt::UniqueConnection);
connect(NetworkManager::notifier(), SIGNAL(statusChanged(NetworkManager::Status)),
SLOT(statusChanged(NetworkManager::Status)), Qt::UniqueConnection);
}
void NetworkModel::initializeSignals(const NetworkManager::ActiveConnection::Ptr& activeConnection)
{
if (activeConnection->vpn()) {
NetworkManager::VpnConnection::Ptr vpnConnection = activeConnection.objectCast<NetworkManager::VpnConnection>();
if (vpnConnection) {
connect(vpnConnection.data(), SIGNAL(stateChanged(NetworkManager::VpnConnection::State,NetworkManager::VpnConnection::StateChangeReason)),
SLOT(activeVpnConnectionStateChanged(NetworkManager::VpnConnection::State,NetworkManager::VpnConnection::StateChangeReason)), Qt::UniqueConnection);
}
} else {
connect(activeConnection.data(), SIGNAL(stateChanged(NetworkManager::ActiveConnection::State)),
SLOT(activeConnectionStateChanged(NetworkManager::ActiveConnection::State)), Qt::UniqueConnection);
}
}
void NetworkModel::initializeSignals(const NetworkManager::Connection::Ptr& connection)
{
connect(connection.data(), SIGNAL(updated()), SLOT(connectionUpdated()), Qt::UniqueConnection);
}
void NetworkModel::initializeSignals(const NetworkManager::Device::Ptr& device)
{
connect(device.data(), SIGNAL(availableConnectionAppeared(QString)),
SLOT(availableConnectionAppeared(QString)), Qt::UniqueConnection);
connect(device.data(), SIGNAL(availableConnectionDisappeared(QString)),
SLOT(availableConnectionDisappeared(QString)), Qt::UniqueConnection);
connect(device.data(), SIGNAL(ipV4ConfigChanged()),
SLOT(ipConfigChanged()), Qt::UniqueConnection);
connect(device.data(), SIGNAL(ipV6ConfigChanged()),
SLOT(ipConfigChanged()), Qt::UniqueConnection);
connect(device.data(), SIGNAL(ipInterfaceChanged()),
SLOT(ipInterfaceChanged()));
connect(device.data(), SIGNAL(stateChanged(NetworkManager::Device::State,NetworkManager::Device::State,NetworkManager::Device::StateChangeReason)),
SLOT(deviceStateChanged(NetworkManager::Device::State,NetworkManager::Device::State,NetworkManager::Device::StateChangeReason)), Qt::UniqueConnection);
if (device->type() == NetworkManager::Device::Wifi) {
NetworkManager::WirelessDevice::Ptr wifiDev = device.objectCast<NetworkManager::WirelessDevice>();
connect(wifiDev.data(), SIGNAL(networkAppeared(QString)),
SLOT(wirelessNetworkAppeared(QString)), Qt::UniqueConnection);
connect(wifiDev.data(), SIGNAL(networkDisappeared(QString)),
SLOT(wirelessNetworkDisappeared(QString)), Qt::UniqueConnection);
}
if (device->type() == NetworkManager::Device::Wimax) {
NetworkManager::WimaxDevice::Ptr wimaxDev = device.objectCast<NetworkManager::WimaxDevice>();
connect(wimaxDev.data(), SIGNAL(nspAppeared(QString)),
SLOT(wimaxNspAppeared(QString)));
connect(wimaxDev.data(), SIGNAL(nspDisappeared(QString)),
SLOT(wimaxNspDisappeared(QString)));
}
#if WITH_MODEMMANAGER_SUPPORT
else if (device->type() == NetworkManager::Device::Modem) {
NetworkManager::ModemDevice::Ptr modemDev = device.objectCast<NetworkManager::ModemDevice>();
ModemManager::Modem::Ptr modemNetwork = modemDev->getModemNetworkIface();
if (modemDev->isValid()) {
connect(modemNetwork.data(), SIGNAL(signalQualityChanged(uint)),
SLOT(gsmNetworkSignalQualityChanged(uint)), Qt::UniqueConnection);
connect(modemNetwork.data(), SIGNAL(accessTechnologyChanged(ModemManager::Modem::AccessTechnologies)),
SLOT(gsmNetworkAccessTechnologyChanged(ModemManager::Modem::AccessTechnologies)), Qt::UniqueConnection);
connect(modemNetwork.data(), SIGNAL(currentModesChanged()),
SLOT(gsmNetworkCurrentModesChanged()), Qt::UniqueConnection);
}
}
#endif
}
void NetworkModel::initializeSignals(const NetworkManager::WimaxNsp::Ptr& nsp)
{
connect(nsp.data(), SIGNAL(signalQualityChanged(uint)),
SLOT(wimaxNspSignalChanged(uint)));
}
void NetworkModel::initializeSignals(const NetworkManager::WirelessNetwork::Ptr& network)
{
connect(network.data(), SIGNAL(signalStrengthChanged(int)),
SLOT(wirelessNetworkSignalChanged(int)), Qt::UniqueConnection);
connect(network.data(), SIGNAL(referenceAccessPointChanged(QString)),
SLOT(wirelessNetworkReferenceApChanged(QString)), Qt::UniqueConnection);
}
void NetworkModel::addActiveConnection(const NetworkManager::ActiveConnection::Ptr& activeConnection)
{
initializeSignals(activeConnection);
NetworkManager::Device::Ptr device;
NetworkManager::Connection::Ptr connection = activeConnection->connection();
// Not necessary to have device for VPN connections
if (activeConnection && !activeConnection->vpn() && !activeConnection->devices().isEmpty()) {
device = NetworkManager::findNetworkInterface(activeConnection->devices().first());
}
// Check whether we have a base connection
if (!m_list.contains(NetworkItemsList::Uuid, connection->uuid())) {
// Active connection appeared before a base connection, so we have to add its base connection first
addConnection(connection);
}
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::NetworkItemsList::Uuid, connection->uuid())) {
if (((device && device->uni() == item->devicePath()) || item->devicePath().isEmpty()) || item->type() == NetworkManager::ConnectionSettings::Vpn) {
item->setActiveConnectionPath(activeConnection->path());
item->setConnectionState(activeConnection->state());
if (activeConnection->vpn()) {
NetworkManager::VpnConnection::Ptr vpnConnection = activeConnection.objectCast<NetworkManager::VpnConnection>();
NetworkManager::VpnConnection::State state = vpnConnection->state();
if (state == NetworkManager::VpnConnection::Prepare ||
state == NetworkManager::VpnConnection::NeedAuth ||
state == NetworkManager::VpnConnection::Connecting ||
state == NetworkManager::VpnConnection::GettingIpConfig) {
item->setConnectionState(NetworkManager::ActiveConnection::Activating);
} else if (state == NetworkManager::VpnConnection::Activated) {
item->setConnectionState(NetworkManager::ActiveConnection::Activated);
} else {
item->setConnectionState(NetworkManager::ActiveConnection::Deactivated);
}
item->setVpnState(state);
}
updateItem(item);
nmDebug() << "Item " << item->name() << ": active connection state changed to " << item->connectionState();
}
}
}
void NetworkModel::addAvailableConnection(const QString& connection, const NetworkManager::Device::Ptr& device)
{
checkAndCreateDuplicate(connection, device);
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Connection, connection)) {
// The item is already associated with another device
if (!item->devicePath().isEmpty()) {
continue;
}
if (device->ipInterfaceName().isEmpty()) {
item->setDeviceName(device->interfaceName());
} else {
item->setDeviceName(device->ipInterfaceName());
}
item->setDevicePath(device->uni());
item->setDeviceState(device->state());
nmDebug() << "Item " << item->name() << ": device changed to " << item->devicePath();
#if WITH_MODEMMANAGER_SUPPORT
if (device->type() == NetworkManager::Device::Modem) {
ModemManager::ModemDevice::Ptr modemDevice = ModemManager::findModemDevice(device->udi());
if (modemDevice) {
ModemManager::Modem::Ptr modemInterface = modemDevice->interface(ModemManager::ModemDevice::ModemInterface).objectCast<ModemManager::Modem>();
if (modemInterface) {
item->setSignal(modemInterface->signalQuality().signal);
nmDebug() << "Item " << item->name() << ": signal changed to " << item->signal();
}
}
}
#endif
if (item->type() == NetworkManager::ConnectionSettings::Wireless && item->mode() == NetworkManager::WirelessSetting::Infrastructure) {
// Find an accesspoint which could be removed, because it will be merged with a connection
foreach (NetworkModelItem * secondItem, m_list.returnItems(NetworkItemsList::Ssid, item->ssid())) {
if (secondItem->itemType() == NetworkModelItem::AvailableAccessPoint && secondItem->devicePath() == item->devicePath()) {
const int row = m_list.indexOf(secondItem);
nmDebug() << "Access point " << secondItem->name() << ": merged to " << item->name() << " connection";
if (row >= 0) {
beginRemoveRows(QModelIndex(), row, row);
m_list.removeItem(secondItem);
secondItem->deleteLater();
endRemoveRows();
}
break;
}
}
NetworkManager::WirelessDevice::Ptr wifiDevice = device.objectCast<NetworkManager::WirelessDevice>();
if (wifiDevice) {
NetworkManager::WirelessNetwork::Ptr wifiNetwork = wifiDevice->findNetwork(item->ssid());
if (wifiNetwork) {
updateFromWirelessNetwork(item, wifiNetwork);
}
}
}
if (item->type() == NetworkManager::ConnectionSettings::Wimax) {
// Find a nsp which could be removed, because it will be merged with a connection
foreach (NetworkModelItem * secondItem, m_list.returnItems(NetworkItemsList::Nsp, item->nsp())) {
if (secondItem->itemType() == NetworkModelItem::AvailableNsp && secondItem->devicePath() == item->devicePath()) {
const int row = m_list.indexOf(secondItem);
nmDebug() << "Nsp " << secondItem->name() << ": merged to " << item->name() << " connection";
if (row >= 0) {
beginRemoveRows(QModelIndex(), row, row);
m_list.removeItem(secondItem);
secondItem->deleteLater();
endRemoveRows();
}
break;
}
}
NetworkManager::WimaxDevice::Ptr wimaxDevice = device.objectCast<NetworkManager::WimaxDevice>();
NetworkManager::WimaxNsp::Ptr nsp = wimaxDevice->findNsp(item->specificPath());
if (nsp) {
updateFromWimaxNsp(item, nsp);
}
}
updateItem(item);
break;
}
}
void NetworkModel::addConnection(const NetworkManager::Connection::Ptr& connection)
{
// Can't add a connection without name or uuid
if (connection->name().isEmpty() || connection->uuid().isEmpty()) {
return;
}
initializeSignals(connection);
NetworkManager::ConnectionSettings::Ptr settings = connection->settings();
NetworkManager::WimaxSetting::Ptr wimaxSetting;
NetworkManager::WirelessSetting::Ptr wirelessSetting;
if (settings->connectionType() == NetworkManager::ConnectionSettings::Wireless) {
wirelessSetting = settings->setting(NetworkManager::Setting::Wireless).dynamicCast<NetworkManager::WirelessSetting>();
}
if (settings->connectionType() == NetworkManager::ConnectionSettings::Wimax) {
wimaxSetting = settings->setting(NetworkManager::Setting::Wimax).dynamicCast<NetworkManager::WimaxSetting>();
}
// Check whether the connection is already in the model to avoid duplicates, but this shouldn't happen
if (!m_list.contains(NetworkItemsList::Connection, connection->path())) {
NetworkModelItem * item = new NetworkModelItem();
item->setConnectionPath(connection->path());
item->setName(settings->id());
item->setTimestamp(settings->timestamp());
item->setType(settings->connectionType());
item->setUuid(settings->uuid());
item->setSlave(settings->isSlave());
if (item->type() == NetworkManager::ConnectionSettings::Wireless) {
item->setMode(wirelessSetting->mode());
item->setSecurityType(NetworkManager::Utils::securityTypeFromConnectionSetting(settings));
item->setSsid(QString::fromUtf8(wirelessSetting->ssid()));
}
if (item->type() == NetworkManager::ConnectionSettings::Wimax) {
item->setNsp(wimaxSetting->networkName());
}
item->updateDetails();
const int index = m_list.count();
beginInsertRows(QModelIndex(), index, index);
m_list.insertItem(item);
endInsertRows();
nmDebug() << "New connection " << item->name() << " added";
}
}
void NetworkModel::addDevice(const NetworkManager::Device::Ptr& device)
{
initializeSignals(device);
if (device->type() == NetworkManager::Device::Wifi) {
NetworkManager::WirelessDevice::Ptr wifiDev = device.objectCast<NetworkManager::WirelessDevice>();
foreach (const NetworkManager::WirelessNetwork::Ptr& wifiNetwork, wifiDev->networks()) {
addWirelessNetwork(wifiNetwork, wifiDev);
}
}
if (device->type() == NetworkManager::Device::Wimax) {
NetworkManager::WimaxDevice::Ptr wimaxDev = device.objectCast<NetworkManager::WimaxDevice>();
foreach (const QString & nsp, wimaxDev->nsps()) {
NetworkManager::WimaxNsp::Ptr wimaxNsp = wimaxDev->findNsp(nsp);
if (wimaxNsp) {
addWimaxNsp(wimaxNsp, wimaxDev);
}
}
}
foreach (const NetworkManager::Connection::Ptr & connection, device->availableConnections()) {
addAvailableConnection(connection->path(), device);
}
}
void NetworkModel::addWimaxNsp(const NetworkManager::WimaxNsp::Ptr& nsp, const NetworkManager::WimaxDevice::Ptr& device)
{
initializeSignals(nsp);
NetworkModelItem * item = new NetworkModelItem();
if (device->ipInterfaceName().isEmpty()) {
item->setDeviceName(device->interfaceName());
} else {
item->setDeviceName(device->ipInterfaceName());
}
item->setDevicePath(device->uni());
item->setName(nsp->name());
item->setSignal(nsp->signalQuality());
item->setSpecificPath(nsp->uni());
item->setType(NetworkManager::ConnectionSettings::Wimax);
item->updateDetails();
const int index = m_list.count();
beginInsertRows(QModelIndex(), index, index);
m_list.insertItem(item);
endInsertRows();
nmDebug() << "New wireless network " << item->name() << " added";}
void NetworkModel::addWirelessNetwork(const NetworkManager::WirelessNetwork::Ptr& network, const NetworkManager::WirelessDevice::Ptr& device)
{
initializeSignals(network);
NetworkManager::WirelessSetting::NetworkMode mode = NetworkManager::WirelessSetting::Infrastructure;
NetworkManager::Utils::WirelessSecurityType securityType = NetworkManager::Utils::Unknown;
NetworkManager::AccessPoint::Ptr ap = network->referenceAccessPoint();
if (ap && ap->capabilities() & NetworkManager::AccessPoint::Privacy) {
securityType = NetworkManager::Utils::findBestWirelessSecurity(device->wirelessCapabilities(), true, (device->mode() == NetworkManager::WirelessDevice::Adhoc),
ap->capabilities(), ap->wpaFlags(), ap->rsnFlags());
if (network->referenceAccessPoint()->mode() == NetworkManager::AccessPoint::Infra) {
mode = NetworkManager::WirelessSetting::Infrastructure;
} else if (network->referenceAccessPoint()->mode() == NetworkManager::AccessPoint::Adhoc) {
mode = NetworkManager::WirelessSetting::Adhoc;
} else if (network->referenceAccessPoint()->mode() == NetworkManager::AccessPoint::ApMode) {
mode = NetworkManager::WirelessSetting::Ap;
}
}
NetworkModelItem * item = new NetworkModelItem();
if (device->ipInterfaceName().isEmpty()) {
item->setDeviceName(device->interfaceName());
} else {
item->setDeviceName(device->ipInterfaceName());
}
item->setDevicePath(device->uni());
item->setMode(mode);
item->setName(network->ssid());
item->setSignal(network->signalStrength());
item->setSpecificPath(network->referenceAccessPoint()->uni());
item->setSsid(network->ssid());
item->setType(NetworkManager::ConnectionSettings::Wireless);
item->setSecurityType(securityType);
item->updateDetails();
const int index = m_list.count();
beginInsertRows(QModelIndex(), index, index);
m_list.insertItem(item);
endInsertRows();
nmDebug() << "New wireless network " << item->name() << " added";
}
void NetworkModel::checkAndCreateDuplicate(const QString& connection, const NetworkManager::Device::Ptr& device)
{
bool createDuplicate = false;
NetworkModelItem * originalItem = 0;
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Connection, connection)) {
if (!item->duplicate()) {
originalItem = item;
}
if (!item->duplicate() && item->itemType() == NetworkModelItem::AvailableConnection && (item->devicePath() != device->uni() && !item->devicePath().isEmpty())) {
createDuplicate = true;
}
}
if (createDuplicate) {
NetworkModelItem * duplicatedItem = new NetworkModelItem(originalItem);
duplicatedItem->updateDetails();
const int index = m_list.count();
beginInsertRows(QModelIndex(), index, index);
m_list.insertItem(duplicatedItem);
endInsertRows();
}
}
void NetworkModel::onItemUpdated()
{
NetworkModelItem * item = static_cast<NetworkModelItem*>(sender());
if (item) {
updateItem(item);
}
}
void NetworkModel::updateItem(NetworkModelItem * item)
{
const int row = m_list.indexOf(item);
if (row >= 0) {
item->updateDetails();
QModelIndex index = createIndex(row, 0);
emit dataChanged(index, index);
}
}
void NetworkModel::updateItems()
{
foreach (NetworkModelItem * item, m_list.items()) {
updateItem(item);
}
}
void NetworkModel::accessPointSignalStrengthChanged(int signal)
{
NetworkManager::AccessPoint * apPtr = qobject_cast<NetworkManager::AccessPoint*>(sender());
if (apPtr) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Ssid, apPtr->ssid())) {
if (item->specificPath() == apPtr->uni()) {
item->setSignal(signal);
updateItem(item);
nmDebug() << "AccessPoint " << item->name() << ": signal changed to " << item->signal();
}
}
}
}
void NetworkModel::activeConnectionAdded(const QString& activeConnection)
{
NetworkManager::ActiveConnection::Ptr activeCon = NetworkManager::findActiveConnection(activeConnection);
if (activeCon) {
addActiveConnection(activeCon);
}
}
void NetworkModel::activeConnectionRemoved(const QString& activeConnection)
{
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::ActiveConnection, activeConnection)) {
item->setActiveConnectionPath(QString());
item->setConnectionState(NetworkManager::ActiveConnection::Deactivated);
item->setVpnState(NetworkManager::VpnConnection::Disconnected);
updateItem(item);
nmDebug() << "Item " << item->name() << ": active connection removed";
}
}
void NetworkModel::activeConnectionStateChanged(NetworkManager::ActiveConnection::State state)
{
NetworkManager::ActiveConnection * activePtr = qobject_cast<NetworkManager::ActiveConnection*>(sender());
if (activePtr) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::ActiveConnection, activePtr->path())) {
item->setConnectionState(state);
updateItem(item);
nmDebug() << "Item " << item->name() << ": active connection changed to " << item->connectionState();
}
}
}
void NetworkModel::activeVpnConnectionStateChanged(NetworkManager::VpnConnection::State state, NetworkManager::VpnConnection::StateChangeReason reason)
{
Q_UNUSED(reason)
NetworkManager::ActiveConnection *activePtr = qobject_cast<NetworkManager::ActiveConnection*>(sender());
if (activePtr) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::ActiveConnection, activePtr->path())) {
if (state == NetworkManager::VpnConnection::Prepare ||
state == NetworkManager::VpnConnection::NeedAuth ||
state == NetworkManager::VpnConnection::Connecting ||
state == NetworkManager::VpnConnection::GettingIpConfig) {
item->setConnectionState(NetworkManager::ActiveConnection::Activating);
} else if (state == NetworkManager::VpnConnection::Activated) {
item->setConnectionState(NetworkManager::ActiveConnection::Activated);
} else {
item->setConnectionState(NetworkManager::ActiveConnection::Deactivated);
}
item->setVpnState(state);
updateItem(item);
nmDebug() << "Item " << item->name() << ": active connection changed to " << item->connectionState();
}
}
}
void NetworkModel::availableConnectionAppeared(const QString& connection)
{
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
addAvailableConnection(connection, device);
}
void NetworkModel::availableConnectionDisappeared(const QString& connection)
{
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Connection, connection)) {
bool available = false;
QString devicePath = item->devicePath();
QString specificPath = item->specificPath();
// We have to check whether the connection is still available, because it might be
// presented in the model for more devices and we don't want to remove it for all of them.
// Check whether the device is still available
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(devicePath);
if (device) {
// Check whether the connection is still listed as available
foreach (const NetworkManager::Connection::Ptr & connection, device->availableConnections()) {
if (connection->path() == item->connectionPath()) {
available = true;
break;
}
}
}
if (!available) {
item->setDeviceName(QString());
item->setDevicePath(QString());
item->setDeviceState(NetworkManager::Device::UnknownState);
item->setSignal(0);
item->setSpecificPath(QString());
nmDebug() << "Item " << item->name() << " removed as available connection";
// Check whether the connection is still available as an access point, this happens
// when we change its properties, like ssid, bssid, security etc.
if (item->type() == NetworkManager::ConnectionSettings::Wireless && !specificPath.isEmpty()) {
if (device && device->type() == NetworkManager::Device::Wifi) {
NetworkManager::WirelessDevice::Ptr wifiDevice = device.objectCast<NetworkManager::WirelessDevice>();
if (wifiDevice) {
NetworkManager::AccessPoint::Ptr ap = wifiDevice->findAccessPoint(specificPath);
if (ap) {
NetworkManager::WirelessNetwork::Ptr network = wifiDevice->findNetwork(ap->ssid());
if (network) {
addWirelessNetwork(network, wifiDevice);
}
}
}
}
}
// Same for Wimax
if (item->type() == NetworkManager::ConnectionSettings::Wimax && !specificPath.isEmpty()) {
if (device && device->type() == NetworkManager::Device::Wimax) {
NetworkManager::WimaxDevice::Ptr wimaxDevice = device.objectCast<NetworkManager::WimaxDevice>();
if (wimaxDevice) {
NetworkManager::WimaxNsp::Ptr nsp = wimaxDevice->findNsp(specificPath);
if (nsp) {
addWimaxNsp(nsp, wimaxDevice);
}
}
}
}
if (item->duplicate()) {
const int row = m_list.indexOf(item);
if (row >= 0) {
nmDebug() << "Duplicate item " << item->name() << " removed completely";
beginRemoveRows(QModelIndex(), row, row);
m_list.removeItem(item);
item->deleteLater();
endRemoveRows();
}
} else {
updateItem(item);
}
}
available = false;
}
}
void NetworkModel::connectionAdded(const QString& connection)
{
NetworkManager::Connection::Ptr newConnection = NetworkManager::findConnection(connection);
if (newConnection) {
addConnection(newConnection);
}
}
void NetworkModel::connectionRemoved(const QString& connection)
{
bool remove = false;
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Connection, connection)) {
// When the item type is wireless, we can remove only the connection and leave it as an available access point
if (item->type() == NetworkManager::ConnectionSettings::Wireless && !item->devicePath().isEmpty()) {
foreach (NetworkModelItem * secondItem, m_list.items()) {
// Remove it entirely when there is another connection with the same configuration and for the same device
// or it's a shared connection
if ((item->mode() != NetworkManager::WirelessSetting::Infrastructure) ||
(item->connectionPath() != secondItem->connectionPath() &&
item->devicePath() == secondItem->devicePath() &&
item->mode() == secondItem->mode() &&
item->securityType() == secondItem->securityType() &&
item->ssid() == secondItem->ssid())) {
remove = true;
}
}
if (!remove) {
item->setConnectionPath(QString());
item->setName(item->ssid());
item->setSlave(false);
item->setTimestamp(QDateTime());
item->setUuid(QString());
updateItem(item);
nmDebug() << "Item " << item->name() << ": connection removed";
}
} else {
remove = true;
}
if (remove) {
const int row = m_list.indexOf(item);
if (row >= 0) {
nmDebug() << "Item " << item->name() << " removed completely";
beginRemoveRows(QModelIndex(), row, row);
m_list.removeItem(item);
item->deleteLater();
endRemoveRows();
}
}
remove = false;
}
}
void NetworkModel::connectionUpdated()
{
NetworkManager::Connection * connectionPtr = qobject_cast<NetworkManager::Connection*>(sender());
if (connectionPtr) {
NetworkManager::ConnectionSettings::Ptr settings = connectionPtr->settings();
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Connection, connectionPtr->path())) {
item->setConnectionPath(connectionPtr->path());
item->setName(settings->id());
item->setTimestamp(settings->timestamp());
item->setType(settings->connectionType());
item->setUuid(settings->uuid());
if (item->type() == NetworkManager::ConnectionSettings::Wireless) {
NetworkManager::WirelessSetting::Ptr wirelessSetting;
wirelessSetting = settings->setting(NetworkManager::Setting::Wireless).dynamicCast<NetworkManager::WirelessSetting>();
item->setMode(wirelessSetting->mode());
item->setSecurityType(NetworkManager::Utils::securityTypeFromConnectionSetting(settings));
item->setSsid(QString::fromUtf8(wirelessSetting->ssid()));
// TODO check whether BSSID has changed and update the wireless info
}
if (item->type() == NetworkManager::ConnectionSettings::Wimax) {
NetworkManager::WimaxSetting::Ptr wimaxSetting;
wimaxSetting = settings->setting(NetworkManager::Setting::Wimax).dynamicCast<NetworkManager::WimaxSetting>();
item->setNsp(wimaxSetting->networkName());
}
updateItem(item);
nmDebug() << "Item " << item->name() << ": connection updated";
}
}
}
void NetworkModel::deviceAdded(const QString& device)
{
NetworkManager::Device::Ptr dev = NetworkManager::findNetworkInterface(device);
if (dev) {
addDevice(dev);
}
}
void NetworkModel::deviceRemoved(const QString& device)
{
// Make all items unavailable
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, device)) {
availableConnectionDisappeared(item->connectionPath());
}
}
void NetworkModel::deviceStateChanged(NetworkManager::Device::State state, NetworkManager::Device::State oldState, NetworkManager::Device::StateChangeReason reason)
{
Q_UNUSED(oldState);
Q_UNUSED(reason);
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
if (device) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, device->uni())) {
item->setDeviceState(state);
updateItem(item);
// nmDebug() << "Item " << item->name() << ": device state changed to " << item->deviceState();
}
}
}
#if WITH_MODEMMANAGER_SUPPORT
void NetworkModel::gsmNetworkAccessTechnologyChanged(ModemManager::Modem::AccessTechnologies technology)
{
Q_UNUSED(technology);
ModemManager::Modem * gsmNetwork = qobject_cast<ModemManager::Modem*>(sender());
if (gsmNetwork) {
foreach (const NetworkManager::Device::Ptr & dev, NetworkManager::networkInterfaces()) {
if (dev->type() == NetworkManager::Device::Modem) {
NetworkManager::ModemDevice::Ptr modem = dev.objectCast<NetworkManager::ModemDevice>();
if (modem) {
if (modem->getModemNetworkIface()->device() == gsmNetwork->device()) {
// TODO store access technology internally?
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, modem->uni())) {
updateItem(item);
}
}
}
}
}
}
}
void NetworkModel::gsmNetworkCurrentModesChanged()
{
ModemManager::Modem * gsmNetwork = qobject_cast<ModemManager::Modem*>(sender());
if (gsmNetwork) {
foreach (const NetworkManager::Device::Ptr & dev, NetworkManager::networkInterfaces()) {
if (dev->type() == NetworkManager::Device::Modem) {
NetworkManager::ModemDevice::Ptr modem = dev.objectCast<NetworkManager::ModemDevice>();
if (modem) {
if (modem->getModemNetworkIface()->device() == gsmNetwork->device()) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, modem->uni())) {
updateItem(item);
}
}
}
}
}
}
}
void NetworkModel::gsmNetworkSignalQualityChanged(uint signal)
{
ModemManager::Modem * gsmNetwork = qobject_cast<ModemManager::Modem*>(sender());
if (gsmNetwork) {
foreach (const NetworkManager::Device::Ptr & dev, NetworkManager::networkInterfaces()) {
if (dev->type() == NetworkManager::Device::Modem) {
NetworkManager::ModemDevice::Ptr modem = dev.objectCast<NetworkManager::ModemDevice>();
if (modem) {
if (modem->getModemNetworkIface()->device() == gsmNetwork->device()) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, modem->uni())) {
item->setSignal(signal);
updateItem(item);
}
}
}
}
}
}
}
#endif
void NetworkModel::ipConfigChanged()
{
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
if (device) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, device->uni())) {
updateItem(item);
// nmDebug() << "Item " << item->name() << ": device ipconfig changed";
}
}
}
void NetworkModel::ipInterfaceChanged()
{
NetworkManager::Device * device = qobject_cast<NetworkManager::Device*>(sender());
if (device) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Device, device->uni())) {
if (device->ipInterfaceName().isEmpty()) {
item->setDeviceName(device->interfaceName());
} else {
item->setDeviceName(device->ipInterfaceName());
}
}
}
}
void NetworkModel::statusChanged(NetworkManager::Status status)
{
Q_UNUSED(status);
nmDebug() << "NetworkManager state changed to " << status;
// This has probably effect only for VPN connections
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Type, NetworkManager::ConnectionSettings::Vpn)) {
updateItem(item);
}
}
void NetworkModel::wimaxNspAppeared(const QString& nsp)
{
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
if (device && device->type() == NetworkManager::Device::Wimax) {
NetworkManager::WimaxDevice::Ptr wimaxDevice = device.objectCast<NetworkManager::WimaxDevice>();
NetworkManager::WimaxNsp::Ptr wimaxNsp = wimaxDevice->findNsp(nsp);
if (wimaxNsp) {
addWimaxNsp(wimaxNsp, wimaxDevice);
}
}
}
void NetworkModel::wimaxNspDisappeared(const QString& nsp)
{
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
if (device) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Nsp, nsp, device->uni())) {
// Remove the entire item, because it's only NSP or it's a duplicated available connection
if (item->itemType() == NetworkModelItem::AvailableNsp || item->duplicate()) {
const int row = m_list.indexOf(item);
if (row >= 0) {
nmDebug() << "Wimax Nsp " << item->name() << " removed completely";
beginRemoveRows(QModelIndex(), row, row);
m_list.removeItem(item);
item->deleteLater();
endRemoveRows();
}
// Remove only NSP and device from the item and leave it as an unavailable connection
} else {
item->setDeviceName(QString());
item->setDevicePath(QString());
item->setSpecificPath(QString());
item->setSignal(0);
updateItem(item);
nmDebug() << "Item " << item->name() << ": wimax Nsp removed";
}
}
}
}
void NetworkModel::wimaxNspSignalChanged(uint signal)
{
NetworkManager::WimaxNsp * nspPtr = qobject_cast<NetworkManager::WimaxNsp*>(sender());
if (nspPtr) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Nsp, nspPtr->name())) {
if (item->specificPath() == nspPtr->uni()) {
item->setSignal(signal);
updateItem(item);
// nmDebug() << "Wimax Nsp " << item->name() << ": signal changed to " << item->signal();
}
}
}
}
void NetworkModel::wirelessNetworkAppeared(const QString& ssid)
{
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
if (device && device->type() == NetworkManager::Device::Wifi) {
NetworkManager::WirelessDevice::Ptr wirelessDevice = device.objectCast<NetworkManager::WirelessDevice>();
NetworkManager::WirelessNetwork::Ptr network = wirelessDevice->findNetwork(ssid);
addWirelessNetwork(network, wirelessDevice);
}
}
void NetworkModel::wirelessNetworkDisappeared(const QString& ssid)
{
NetworkManager::Device::Ptr device = NetworkManager::findNetworkInterface(qobject_cast<NetworkManager::Device*>(sender())->uni());
if (device) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Ssid, ssid, device->uni())) {
// Remove the entire item, because it's only AP or it's a duplicated available connection
if (item->itemType() == NetworkModelItem::AvailableAccessPoint || item->duplicate()) {
const int row = m_list.indexOf(item);
if (row >= 0) {
nmDebug() << "Wireless network " << item->name() << " removed completely";
beginRemoveRows(QModelIndex(), row, row);
m_list.removeItem(item);
item->deleteLater();
endRemoveRows();
}
// Remove only AP and device from the item and leave it as an unavailable connection
} else {
if (item->mode() == NetworkManager::WirelessSetting::Infrastructure) {
item->setDeviceName(QString());
item->setDevicePath(QString());
item->setSpecificPath(QString());
}
item->setSignal(0);
updateItem(item);
nmDebug() << "Item " << item->name() << ": wireless network removed";
}
}
}
}
void NetworkModel::wirelessNetworkReferenceApChanged(const QString& accessPoint)
{
NetworkManager::WirelessNetwork * networkPtr = qobject_cast<NetworkManager::WirelessNetwork*>(sender());
if (networkPtr) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Ssid, networkPtr->ssid(), networkPtr->device())) {
NetworkManager::Connection::Ptr connection = NetworkManager::findConnection(item->connectionPath());
if (connection) {
NetworkManager::WirelessSetting::Ptr wirelessSetting = connection->settings()->setting(NetworkManager::Setting::Wireless).staticCast<NetworkManager::WirelessSetting>();
if (wirelessSetting) {
if (wirelessSetting->bssid().isEmpty()) {
item->setSpecificPath(accessPoint);
updateItem(item);
}
}
}
}
}
}
void NetworkModel::wirelessNetworkSignalChanged(int signal)
{
NetworkManager::WirelessNetwork * networkPtr = qobject_cast<NetworkManager::WirelessNetwork*>(sender());
if (networkPtr) {
foreach (NetworkModelItem * item, m_list.returnItems(NetworkItemsList::Ssid, networkPtr->ssid(), networkPtr->device())) {
if (item->specificPath() == networkPtr->referenceAccessPoint()->uni()) {
item->setSignal(signal);
updateItem(item);
nmDebug() << "Wireless network " << item->name() << ": signal changed to " << item->signal();
}
}
}
}
NetworkManager::Utils::WirelessSecurityType NetworkModel::alternativeWirelessSecurity(const NetworkManager::Utils::WirelessSecurityType type)
{
if (type == NetworkManager::Utils::WpaPsk) {
return NetworkManager::Utils::Wpa2Psk;
} else if (type == NetworkManager::Utils::WpaEap) {
return NetworkManager::Utils::Wpa2Eap;
} else if (type == NetworkManager::Utils::Wpa2Psk) {
return NetworkManager::Utils::WpaPsk;
} else if (type == NetworkManager::Utils::Wpa2Eap) {
return NetworkManager::Utils::WpaEap;
}
return type;
}
void NetworkModel::updateFromWimaxNsp(NetworkModelItem* item, const NetworkManager::WimaxNsp::Ptr& nsp)
{
item->setSignal(nsp->signalQuality());
item->setSpecificPath(nsp->uni());
}
void NetworkModel::updateFromWirelessNetwork(NetworkModelItem* item, const NetworkManager::WirelessNetwork::Ptr& network)
{
// Check whether the connection is associated with some concrete AP
NetworkManager::Connection::Ptr connection = NetworkManager::findConnection(item->connectionPath());
if (connection) {
NetworkManager::WirelessSetting::Ptr wirelessSetting = connection->settings()->setting(NetworkManager::Setting::Wireless).staticCast<NetworkManager::WirelessSetting>();
if (wirelessSetting) {
if (!wirelessSetting->bssid().isEmpty()) {
foreach (const NetworkManager::AccessPoint::Ptr ap, network->accessPoints()) {
if (ap->hardwareAddress() == NetworkManager::Utils::macAddressAsString(wirelessSetting->bssid())) {
item->setSignal(ap->signalStrength());
item->setSpecificPath(ap->uni());
// We need to watch this AP for signal changes
connect(ap.data(), SIGNAL(signalStrengthChanged(int)), SLOT(accessPointSignalStrengthChanged(int)), Qt::UniqueConnection);
}
}
} else {
item->setSignal(network->signalStrength());
item->setSpecificPath(network->referenceAccessPoint()->uni());
}
}
}
}