// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- const Gio = imports.gi.Gio; const Lang = imports.lang; const Shell = imports.gi.Shell; const Signals = imports.signals; // _getProvidersTable: // // Gets the table of references between MCCMNC and operator name // let _providersTable; function _getProvidersTable() { if (_providersTable) return _providersTable; return _providersTable = Shell.mobile_providers_parse(null,null); } // findProviderForMCCMNC: // @table: a table of country code keys and Shell.CountryMobileProvider values // @needle: operator code, given as MCCMNC string // // Tries to find the operator name corresponding to the given MCCMNC // function findProviderForMCCMNC(table, needle) { let needlemcc = needle.substring(0, 3); let needlemnc = needle.substring(3, needle.length); let name2, name3; for (let iter in table) { let country = table[iter]; let providers = country.get_providers(); // Search through each country's providers for (let i = 0; i < providers.length; i++) { let provider = providers[i]; // Search through MCC/MNC list let list = provider.get_gsm_mcc_mnc(); for (let j = 0; j < list.length; j++) { let mccmnc = list[j]; // Match both 2-digit and 3-digit MNC; prefer a // 3-digit match if found, otherwise a 2-digit one. if (mccmnc.mcc != needlemcc) continue; // MCC was wrong if (!name3 && needle.length == 6 && needlemnc == mccmnc.mnc) name3 = provider.name; if (!name2 && needlemnc.substring(0, 2) == mccmnc.mnc.substring(0, 2)) name2 = provider.name; if (name2 && name3) break; } } } return name3 || name2 || null; } // findOperatorName: // @operator_name: operator name // @operator_code: operator code // // Given an operator name string (which may not be a real operator name) and an // operator code string, tries to find a proper operator name to display. // function findOperatorName(operator_name, operator_code) { if (operator_name) { if (operator_name.length != 0 && (operator_name.length > 6 || operator_name.length < 5)) { // this looks like a valid name, i.e. not an MCCMNC (that some // devices return when not yet connected return operator_name; } if (isNaN(parseInt(operator_name))) { // name is definitely not a MCCMNC, so it may be a name // after all; return that return operator_name; } } let needle; if ((!operator_name || operator_name.length == 0) && operator_code) needle = operator_code; else if (operator_name && (operator_name.length == 6 || operator_name.length == 5)) needle = operator_name; else // nothing to search return null; let table = _getProvidersTable(); return findProviderForMCCMNC(table, needle); } // findProviderForSid: // @table: a table of country code keys and Shell.CountryMobileProvider values // @sid: System Identifier of the serving CDMA network // // Tries to find the operator name corresponding to the given SID // function findProviderForSid(table, sid) { if (sid == 0) return null; // Search through each country for (let iter in table) { let country = table[iter]; let providers = country.get_providers(); // Search through each country's providers for (let i = 0; i < providers.length; i++) { let provider = providers[i]; let cdma_sid = provider.get_cdma_sid(); // Search through CDMA SID list for (let j = 0; j < cdma_sid.length; j++) { if (cdma_sid[j] == sid) return provider.name; } } } return null; } //------------------------------------------------------------------------------ // Support for the old ModemManager interface (MM < 0.7) //------------------------------------------------------------------------------ // The following are not the complete interfaces, just the methods we need // (or may need in the future) const ModemGsmNetworkInterface = ; const ModemGsmNetworkProxy = Gio.DBusProxy.makeProxyWrapper(ModemGsmNetworkInterface); const ModemCdmaInterface = ; const ModemCdmaProxy = Gio.DBusProxy.makeProxyWrapper(ModemCdmaInterface); const ModemGsm = new Lang.Class({ Name: 'ModemGsm', _init: function(path) { this._proxy = new ModemGsmNetworkProxy(Gio.DBus.system, 'org.freedesktop.ModemManager', path); this.signal_quality = 0; this.operator_name = null; // Code is duplicated because the function have different signatures this._proxy.connectSignal('SignalQuality', Lang.bind(this, function(proxy, sender, [quality]) { this.signal_quality = quality; this.emit('notify::signal-quality'); })); this._proxy.connectSignal('RegistrationInfo', Lang.bind(this, function(proxy, sender, [status, code, name]) { this.operator_name = findOperatorName(name, code); this.emit('notify::operator-name'); })); this._proxy.GetRegistrationInfoRemote(Lang.bind(this, function([result], err) { if (err) { log(err); return; } let [status, code, name] = result; this.operator_name = findOperatorName(name, code); this.emit('notify::operator-name'); })); this._proxy.GetSignalQualityRemote(Lang.bind(this, function(result, err) { if (err) { // it will return an error if the device is not connected this.signal_quality = 0; } else { let [quality] = result; this.signal_quality = quality; } this.emit('notify::signal-quality'); })); } }); Signals.addSignalMethods(ModemGsm.prototype); const ModemCdma = new Lang.Class({ Name: 'ModemCdma', _init: function(path) { this._proxy = new ModemCdmaProxy(Gio.DBus.system, 'org.freedesktop.ModemManager', path); this.signal_quality = 0; this.operator_name = null; this._proxy.connectSignal('SignalQuality', Lang.bind(this, function(proxy, sender, params) { this.signal_quality = params[0]; this.emit('notify::signal-quality'); // receiving this signal means the device got activated // and we can finally call GetServingSystem if (this.operator_name == null) this._refreshServingSystem(); })); this._proxy.GetSignalQualityRemote(Lang.bind(this, function(result, err) { if (err) { // it will return an error if the device is not connected this.signal_quality = 0; } else { let [quality] = result; this.signal_quality = quality; } this.emit('notify::signal-quality'); })); }, _refreshServingSystem: function() { this._proxy.GetServingSystemRemote(Lang.bind(this, function([result], err) { if (err) { // it will return an error if the device is not connected this.operator_name = null; } else { let [bandClass, band, id] = result; if (name.length > 0) { let table = _getProvidersTable(); this.operator_name = findProviderForSid(table, id); } else this.operator_name = null; } this.emit('notify::operator-name'); })); } }); Signals.addSignalMethods(ModemCdma.prototype); //------------------------------------------------------------------------------ // Support for the new ModemManager1 interface (MM >= 0.7) //------------------------------------------------------------------------------ const BroadbandModemInterface = ; const BroadbandModemProxy = Gio.DBusProxy.makeProxyWrapper(BroadbandModemInterface); const BroadbandModem3gppInterface = ; const BroadbandModem3gppProxy = Gio.DBusProxy.makeProxyWrapper(BroadbandModem3gppInterface); const BroadbandModemCdmaInterface = ; const BroadbandModemCdmaProxy = Gio.DBusProxy.makeProxyWrapper(BroadbandModemCdmaInterface); const BroadbandModem = new Lang.Class({ Name: 'BroadbandModem', _init: function(path, capabilities) { this._proxy = new BroadbandModemProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); this._proxy_3gpp = new BroadbandModem3gppProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); this._proxy_cdma = new BroadbandModemCdmaProxy(Gio.DBus.system, 'org.freedesktop.ModemManager1', path); this._capabilities = capabilities; this._proxy.connect('g-properties-changed', Lang.bind(this, function(proxy, properties) { if ('SignalQuality' in properties.deep_unpack()) this._reloadSignalQuality(); })); this._reloadSignalQuality(); this._proxy_3gpp.connect('g-properties-changed', Lang.bind(this, function(proxy, properties) { let unpacked = properties.deep_unpack(); if ('OperatorName' in unpacked || 'OperatorCode' in unpacked) this._reload3gppOperatorName(); })); this._reload3gppOperatorName(); this._proxy_cdma.connect('g-properties-changed', Lang.bind(this, function(proxy, properties) { let unpacked = properties.deep_unpack(); if ('Nid' in unpacked || 'Sid' in unpacked) this._reloadCdmaOperatorName(); })); this._reloadCdmaOperatorName(); }, _reloadSignalQuality: function() { let [quality, recent] = this._proxy.SignalQuality; this.signal_quality = quality; this.emit('notify::signal-quality'); }, _reloadOperatorName: function() { let new_name = ""; if (this.operator_name_3gpp && this.operator_name_3gpp.length > 0) new_name += this.operator_name_3gpp; if (this.operator_name_cdma && this.operator_name_cdma.length > 0) { if (new_name != "") new_name += ", "; new_name += this.operator_name_cdma; } this.operator_name = new_name; this.emit('notify::operator-name'); }, _reload3gppOperatorName: function() { let name = this._proxy_3gpp.OperatorName; let code = this._proxy_3gpp.OperatorCode; this.operator_name_3gpp = findOperatorName(name, code); this._reloadOperatorName(); }, _reloadCdmaOperatorName: function() { let sid = this._proxy_cdma.Sid; this.operator_name_cdma = findProviderForSid(sid); this._reloadOperatorName(); } }); Signals.addSignalMethods(BroadbandModem.prototype);