| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287 | // This file is part of the AppIndicator/KStatusNotifierItem GNOME Shell extension//// This program is free software; you can redistribute it and/or// modify it under the terms of the GNU General Public License// as published by the Free Software Foundation; either version 2// of the License, or (at your option) any later version.//// This program 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 General Public License for more details.//// You should have received a copy of the GNU General Public License// along with this program; if not, write to the Free Software// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.import Gio from 'gi://Gio';import GLib from 'gi://GLib';import * as AppIndicator from './appIndicator.js';import * as IndicatorStatusIcon from './indicatorStatusIcon.js';import * as Interfaces from './interfaces.js';import * as PromiseUtils from './promiseUtils.js';import * as Util from './util.js';import * as DBusMenu from './dbusMenu.js';import {DBusProxy} from './dbusProxy.js';// TODO: replace with org.freedesktop and /org/freedesktop when approvedconst KDE_PREFIX = 'org.kde';export const WATCHER_BUS_NAME = `${KDE_PREFIX}.StatusNotifierWatcher`;const WATCHER_OBJECT = '/StatusNotifierWatcher';const DEFAULT_ITEM_OBJECT_PATH = '/StatusNotifierItem';/* * The StatusNotifierWatcher class implements the StatusNotifierWatcher dbus object */export class StatusNotifierWatcher {    constructor(watchDog) {        this._watchDog = watchDog;        this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(Interfaces.StatusNotifierWatcher, this);        try {            this._dbusImpl.export(Gio.DBus.session, WATCHER_OBJECT);        } catch (e) {            Util.Logger.warn(`Failed to export ${WATCHER_OBJECT}`);            logError(e);        }        this._cancellable = new Gio.Cancellable();        this._everAcquiredName = false;        this._ownName = Gio.DBus.session.own_name(WATCHER_BUS_NAME,            Gio.BusNameOwnerFlags.NONE,            this._acquiredName.bind(this),            this._lostName.bind(this));        this._items = new Map();        try {            this._dbusImpl.emit_signal('StatusNotifierHostRegistered', null);        } catch (e) {            Util.Logger.warn(`Failed to notify registered host ${WATCHER_OBJECT}`);        }        this._seekStatusNotifierItems().catch(e => {            if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED))                logError(e, 'Looking for StatusNotifierItem\'s');        });    }    _acquiredName() {        this._everAcquiredName = true;        this._watchDog.nameAcquired = true;    }    _lostName() {        if (this._everAcquiredName)            Util.Logger.debug(`Lost name${WATCHER_BUS_NAME}`);        else            Util.Logger.warn(`Failed to acquire ${WATCHER_BUS_NAME}`);        this._watchDog.nameAcquired = false;    }    async _registerItem(service, busName, objPath) {        const id = Util.indicatorId(service, busName, objPath);        if (this._items.has(id)) {            Util.Logger.warn(`Item ${id} is already registered`);            return;        }        Util.Logger.debug(`Registering StatusNotifierItem ${id}`);        try {            const indicator = new AppIndicator.AppIndicator(service, busName, objPath);            this._items.set(id, indicator);            indicator.connect('destroy', () => this._onIndicatorDestroyed(indicator));            indicator.connect('name-owner-changed', async () => {                if (!indicator.hasNameOwner) {                    try {                        await new PromiseUtils.TimeoutPromise(500,                            GLib.PRIORITY_DEFAULT, this._cancellable);                        if (this._items.has(id) && !indicator.hasNameOwner)                            indicator.destroy();                    } catch (e) {                        if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED))                            logError(e);                    }                }            });            // if the desktop is not ready delay the icon creation and signal emissions            await Util.waitForStartupCompletion(indicator.cancellable);            const statusIcon = new IndicatorStatusIcon.IndicatorStatusIcon(indicator);            IndicatorStatusIcon.addIconToPanel(statusIcon);            this._dbusImpl.emit_signal('StatusNotifierItemRegistered',                GLib.Variant.new('(s)', [indicator.uniqueId]));            this._dbusImpl.emit_property_changed('RegisteredStatusNotifierItems',                GLib.Variant.new('as', this.RegisteredStatusNotifierItems));        } catch (e) {            if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED))                logError(e);            throw e;        }    }    async _ensureItemRegistered(service, busName, objPath) {        const id = Util.indicatorId(service, busName, objPath);        const item = this._items.get(id);        if (item) {            // delete the old one and add the new indicator            Util.Logger.debug(`Attempting to re-register ${id}; resetting instead`);            item.reset();            return;        }        await this._registerItem(service, busName, objPath);    }    async _seekStatusNotifierItems() {        // Some indicators (*coff*, dropbox, *coff*) do not re-register again        // when the plugin is enabled/disabled, thus we need to manually look        // for the objects in the session bus that implements the        // StatusNotifierItem interface... However let's do it after a low        // priority idle, so that it won't affect startup.        const cancellable = this._cancellable;        const bus = Gio.DBus.session;        const uniqueNames = await Util.getBusNames(bus, cancellable);        const introspectName = async name => {            const nodes = Util.introspectBusObject(bus, name, cancellable,                ['org.kde.StatusNotifierItem']);            const services = [...uniqueNames.get(name)];            for await (const node of nodes) {                const {path} = node;                const ids = services.map(s => Util.indicatorId(s, name, path));                if (ids.every(id => !this._items.has(id))) {                    const service = services.find(s =>                        s && s.startsWith('org.kde.StatusNotifierItem')) || services[0];                    const id = Util.indicatorId(                        path === DEFAULT_ITEM_OBJECT_PATH ? service : null,                        name, path);                    Util.Logger.warn(`Using Brute-force mode for StatusNotifierItem ${id}`);                    this._registerItem(service, name, path);                }            }        };        await Promise.allSettled([...uniqueNames.keys()].map(n => introspectName(n)));    }    async RegisterStatusNotifierItemAsync(params, invocation) {        // it would be too easy if all application behaved the same        // instead, ayatana patched gnome apps to send a path        // while kde apps send a bus name        const [service] = params;        let busName, objPath;        if (service.charAt(0) === '/') { // looks like a path            busName = invocation.get_sender();            objPath = service;        } else if (service.match(Util.BUS_ADDRESS_REGEX)) {            try {                busName = await Util.getUniqueBusName(invocation.get_connection(),                    service, this._cancellable);            } catch (e) {                logError(e);            }            objPath = DEFAULT_ITEM_OBJECT_PATH;        }        if (!busName || !objPath) {            const error = `Impossible to register an indicator for parameters '${                service.toString()}'`;            Util.Logger.warn(error);            invocation.return_dbus_error('org.gnome.gjs.JSError.ValueError',                error);            return;        }        try {            await this._ensureItemRegistered(service, busName, objPath);            invocation.return_value(null);        } catch (e) {            if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED))                logError(e);            invocation.return_dbus_error('org.gnome.gjs.JSError.ValueError',                e.message);        }    }    _onIndicatorDestroyed(indicator) {        const {uniqueId} = indicator;        this._items.delete(uniqueId);        try {            this._dbusImpl.emit_signal('StatusNotifierItemUnregistered',                GLib.Variant.new('(s)', [uniqueId]));            this._dbusImpl.emit_property_changed('RegisteredStatusNotifierItems',                GLib.Variant.new('as', this.RegisteredStatusNotifierItems));        } catch (e) {            Util.Logger.warn(`Failed to emit signals: ${e}`);        }    }    RegisterStatusNotifierHostAsync(_service, invocation) {        invocation.return_error_literal(            Gio.DBusError,            Gio.DBusError.NOT_SUPPORTED,            'Registering additional notification hosts is not supported');    }    IsNotificationHostRegistered() {        return true;    }    get RegisteredStatusNotifierItems() {        return Array.from(this._items.values()).map(i => i.uniqueId);    }    get IsStatusNotifierHostRegistered() {        return true;    }    get ProtocolVersion() {        return 0;    }    destroy() {        if (this._isDestroyed)            return;        // this doesn't do any sync operation and doesn't allow us to hook up        // the event of being finished which results in our unholy debounce hack        // (see extension.js)        this._items.forEach(indicator => indicator.destroy());        this._cancellable.cancel();        try {            this._dbusImpl.emit_signal('StatusNotifierHostUnregistered', null);        } catch (e) {            Util.Logger.warn(`Failed to emit uinregistered signal: ${e}`);        }        Gio.DBus.session.unown_name(this._ownName);        try {            this._dbusImpl.unexport();        } catch (e) {            Util.Logger.warn(`Failed to unexport watcher object: ${e}`);        }        DBusMenu.DBusClient.destroy();        AppIndicator.AppIndicatorProxy.destroy();        DBusProxy.destroy();        Util.destroyDefaultTheme();        this._dbusImpl.run_dispose();        delete this._dbusImpl;        delete this._items;        this._isDestroyed = true;    }}
 |