123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592 |
- // 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 Clutter from 'gi://Clutter';
- import Gio from 'gi://Gio';
- import GObject from 'gi://GObject';
- import St from 'gi://St';
- import * as AppDisplay from 'resource:///org/gnome/shell/ui/appDisplay.js';
- import * as Main from 'resource:///org/gnome/shell/ui/main.js';
- import * as Panel from 'resource:///org/gnome/shell/ui/panel.js';
- import * as PanelMenu from 'resource:///org/gnome/shell/ui/panelMenu.js';
- import * as AppIndicator from './appIndicator.js';
- import * as PromiseUtils from './promiseUtils.js';
- import * as SettingsManager from './settingsManager.js';
- import * as Util from './util.js';
- import * as DBusMenu from './dbusMenu.js';
- const DEFAULT_ICON_SIZE = Panel.PANEL_ICON_SIZE || 16;
- export function addIconToPanel(statusIcon) {
- if (!(statusIcon instanceof BaseStatusIcon))
- throw TypeError(`Unexpected icon type: ${statusIcon}`);
- const settings = SettingsManager.getDefaultGSettings();
- const indicatorId = `appindicator-${statusIcon.uniqueId}`;
- const currentIcon = Main.panel.statusArea[indicatorId];
- if (currentIcon) {
- if (currentIcon !== statusIcon)
- currentIcon.destroy();
- Main.panel.statusArea[indicatorId] = null;
- }
- Main.panel.addToStatusArea(indicatorId, statusIcon, 1,
- settings.get_string('tray-pos'));
- Util.connectSmart(settings, 'changed::tray-pos', statusIcon, () =>
- addIconToPanel(statusIcon));
- }
- export function getTrayIcons() {
- return Object.values(Main.panel.statusArea).filter(
- i => i instanceof IndicatorStatusTrayIcon);
- }
- export function getAppIndicatorIcons() {
- return Object.values(Main.panel.statusArea).filter(
- i => i instanceof IndicatorStatusIcon);
- }
- export const BaseStatusIcon = GObject.registerClass(
- class IndicatorBaseStatusIcon extends PanelMenu.Button {
- _init(menuAlignment, nameText, iconActor, dontCreateMenu) {
- super._init(menuAlignment, nameText, dontCreateMenu);
- const settings = SettingsManager.getDefaultGSettings();
- Util.connectSmart(settings, 'changed::icon-opacity', this, this._updateOpacity);
- this.connect('notify::hover', () => this._onHoverChanged());
- if (!super._onDestroy)
- this.connect('destroy', () => this._onDestroy());
- this._box = new St.BoxLayout({style_class: 'panel-status-indicators-box'});
- this.add_child(this._box);
- this._setIconActor(iconActor);
- this._showIfReady();
- }
- _setIconActor(icon) {
- if (!(icon instanceof Clutter.Actor))
- throw new Error(`${icon} is not a valid actor`);
- if (this._icon && this._icon !== icon)
- this._icon.destroy();
- this._icon = icon;
- this._updateEffects();
- this._monitorIconEffects();
- if (this._icon) {
- this._box.add_child(this._icon);
- const id = this._icon.connect('destroy', () => {
- this._icon.disconnect(id);
- this._icon = null;
- this._monitorIconEffects();
- });
- }
- }
- _onDestroy() {
- if (this._icon)
- this._icon.destroy();
- if (super._onDestroy)
- super._onDestroy();
- }
- isReady() {
- throw new GObject.NotImplementedError('isReady() in %s'.format(this.constructor.name));
- }
- get icon() {
- return this._icon;
- }
- get uniqueId() {
- throw new GObject.NotImplementedError('uniqueId in %s'.format(this.constructor.name));
- }
- _showIfReady() {
- this.visible = this.isReady();
- }
- _onHoverChanged() {
- if (this.hover) {
- this.opacity = 255;
- if (this._icon)
- this._icon.remove_effect_by_name('desaturate');
- } else {
- this._updateEffects();
- }
- }
- _updateOpacity() {
- const settings = SettingsManager.getDefaultGSettings();
- const userValue = settings.get_user_value('icon-opacity');
- if (userValue)
- this.opacity = userValue.unpack();
- else
- this.opacity = 255;
- }
- _updateEffects() {
- this._updateOpacity();
- if (this._icon) {
- this._updateSaturation();
- this._updateBrightnessContrast();
- }
- }
- _monitorIconEffects() {
- const settings = SettingsManager.getDefaultGSettings();
- const monitoring = !!this._iconSaturationIds;
- if (!this._icon && monitoring) {
- Util.disconnectSmart(settings, this, this._iconSaturationIds);
- delete this._iconSaturationIds;
- Util.disconnectSmart(settings, this, this._iconBrightnessIds);
- delete this._iconBrightnessIds;
- Util.disconnectSmart(settings, this, this._iconContrastIds);
- delete this._iconContrastIds;
- } else if (this._icon && !monitoring) {
- this._iconSaturationIds =
- Util.connectSmart(settings, 'changed::icon-saturation', this,
- this._updateSaturation);
- this._iconBrightnessIds =
- Util.connectSmart(settings, 'changed::icon-brightness', this,
- this._updateBrightnessContrast);
- this._iconContrastIds =
- Util.connectSmart(settings, 'changed::icon-contrast', this,
- this._updateBrightnessContrast);
- }
- }
- _updateSaturation() {
- const settings = SettingsManager.getDefaultGSettings();
- const desaturationValue = settings.get_double('icon-saturation');
- let desaturateEffect = this._icon.get_effect('desaturate');
- if (desaturationValue > 0) {
- if (!desaturateEffect) {
- desaturateEffect = new Clutter.DesaturateEffect();
- this._icon.add_effect_with_name('desaturate', desaturateEffect);
- }
- desaturateEffect.set_factor(desaturationValue);
- } else if (desaturateEffect) {
- this._icon.remove_effect(desaturateEffect);
- }
- }
- _updateBrightnessContrast() {
- const settings = SettingsManager.getDefaultGSettings();
- const brightnessValue = settings.get_double('icon-brightness');
- const contrastValue = settings.get_double('icon-contrast');
- let brightnessContrastEffect = this._icon.get_effect('brightness-contrast');
- if (brightnessValue !== 0 | contrastValue !== 0) {
- if (!brightnessContrastEffect) {
- brightnessContrastEffect = new Clutter.BrightnessContrastEffect();
- this._icon.add_effect_with_name('brightness-contrast', brightnessContrastEffect);
- }
- brightnessContrastEffect.set_brightness(brightnessValue);
- brightnessContrastEffect.set_contrast(contrastValue);
- } else if (brightnessContrastEffect) {
- this._icon.remove_effect(brightnessContrastEffect);
- }
- }
- });
- /*
- * IndicatorStatusIcon implements an icon in the system status area
- */
- export const IndicatorStatusIcon = GObject.registerClass(
- class IndicatorStatusIcon extends BaseStatusIcon {
- _init(indicator) {
- super._init(0.5, indicator.accessibleName,
- new AppIndicator.IconActor(indicator, DEFAULT_ICON_SIZE));
- this._indicator = indicator;
- this._lastClickTime = -1;
- this._lastClickX = -1;
- this._lastClickY = -1;
- this._box.add_style_class_name('appindicator-box');
- Util.connectSmart(this._indicator, 'ready', this, this._showIfReady);
- Util.connectSmart(this._indicator, 'menu', this, this._updateMenu);
- Util.connectSmart(this._indicator, 'label', this, this._updateLabel);
- Util.connectSmart(this._indicator, 'status', this, this._updateStatus);
- Util.connectSmart(this._indicator, 'reset', this, () => {
- this._updateStatus();
- this._updateLabel();
- });
- Util.connectSmart(this._indicator, 'accessible-name', this, () =>
- this.set_accessible_name(this._indicator.accessibleName));
- Util.connectSmart(this._indicator, 'destroy', this, () => this.destroy());
- this.connect('notify::visible', () => this._updateMenu());
- this._showIfReady();
- }
- _onDestroy() {
- if (this._menuClient) {
- this._menuClient.disconnect(this._menuReadyId);
- this._menuClient.destroy();
- this._menuClient = null;
- }
- super._onDestroy();
- }
- get uniqueId() {
- return this._indicator.uniqueId;
- }
- isReady() {
- return this._indicator && this._indicator.isReady;
- }
- _updateLabel() {
- const {label} = this._indicator;
- if (label) {
- if (!this._label || !this._labelBin) {
- this._labelBin = new St.Bin({
- yAlign: Clutter.ActorAlign.CENTER,
- });
- this._label = new St.Label();
- this._labelBin.add_actor(this._label);
- this._box.add_actor(this._labelBin);
- }
- this._label.set_text(label);
- if (!this._box.contains(this._labelBin))
- this._box.add_actor(this._labelBin); // FIXME: why is it suddenly necessary?
- } else if (this._label) {
- this._labelBin.destroy_all_children();
- this._box.remove_actor(this._labelBin);
- this._labelBin.destroy();
- delete this._labelBin;
- delete this._label;
- }
- }
- _updateStatus() {
- const wasVisible = this.visible;
- this.visible = this._indicator.status !== AppIndicator.SNIStatus.PASSIVE;
- if (this.visible !== wasVisible)
- this._indicator.checkAlive().catch(logError);
- }
- _updateMenu() {
- if (this._menuClient) {
- this._menuClient.disconnect(this._menuReadyId);
- this._menuClient.destroy();
- this._menuClient = null;
- this.menu.removeAll();
- }
- if (this.visible && this._indicator.menuPath) {
- this._menuClient = new DBusMenu.Client(this._indicator.busName,
- this._indicator.menuPath, this._indicator);
- if (this._menuClient.isReady)
- this._menuClient.attachToMenu(this.menu);
- this._menuReadyId = this._menuClient.connect('ready-changed', () => {
- if (this._menuClient.isReady)
- this._menuClient.attachToMenu(this.menu);
- else
- this._updateMenu();
- });
- }
- }
- _showIfReady() {
- if (!this.isReady())
- return;
- this._updateLabel();
- this._updateStatus();
- this._updateMenu();
- }
- _updateClickCount(event) {
- const [x, y] = event.get_coords();
- const time = event.get_time();
- const {doubleClickDistance, doubleClickTime} =
- Clutter.Settings.get_default();
- if (time > (this._lastClickTime + doubleClickTime) ||
- (Math.abs(x - this._lastClickX) > doubleClickDistance) ||
- (Math.abs(y - this._lastClickY) > doubleClickDistance))
- this._clickCount = 0;
- this._lastClickTime = time;
- this._lastClickX = x;
- this._lastClickY = y;
- this._clickCount = (this._clickCount % 2) + 1;
- return this._clickCount;
- }
- _maybeHandleDoubleClick(event) {
- if (this._indicator.supportsActivation === false)
- return Clutter.EVENT_PROPAGATE;
- if (event.get_button() !== Clutter.BUTTON_PRIMARY)
- return Clutter.EVENT_PROPAGATE;
- if (this._updateClickCount(event) === 2) {
- this._indicator.open(...event.get_coords(), event.get_time());
- return Clutter.EVENT_STOP;
- }
- return Clutter.EVENT_PROPAGATE;
- }
- async _waitForDoubleClick() {
- const {doubleClickTime} = Clutter.Settings.get_default();
- this._waitDoubleClickPromise = new PromiseUtils.TimeoutPromise(
- doubleClickTime);
- try {
- await this._waitDoubleClickPromise;
- this.menu.toggle();
- } catch (e) {
- if (!e.matches(Gio.IOErrorEnum, Gio.IOErrorEnum.CANCELLED))
- throw e;
- } finally {
- delete this._waitDoubleClickPromise;
- }
- }
- vfunc_event(event) {
- if (this.menu.numMenuItems && event.type() === Clutter.EventType.TOUCH_BEGIN)
- this.menu.toggle();
- return Clutter.EVENT_PROPAGATE;
- }
- vfunc_button_press_event(event) {
- if (this._waitDoubleClickPromise)
- this._waitDoubleClickPromise.cancel();
- // if middle mouse button clicked send SecondaryActivate dbus event and do not show appindicator menu
- if (event.get_button() === Clutter.BUTTON_MIDDLE) {
- if (Main.panel.menuManager.activeMenu)
- Main.panel.menuManager._closeMenu(true, Main.panel.menuManager.activeMenu);
- this._indicator.secondaryActivate(event.get_time(), ...event.get_coords());
- return Clutter.EVENT_STOP;
- }
- if (event.get_button() === Clutter.BUTTON_SECONDARY) {
- this.menu.toggle();
- return Clutter.EVENT_PROPAGATE;
- }
- const doubleClickHandled = this._maybeHandleDoubleClick(event);
- if (doubleClickHandled === Clutter.EVENT_PROPAGATE &&
- event.get_button() === Clutter.BUTTON_PRIMARY &&
- this.menu.numMenuItems) {
- if (this._indicator.supportsActivation !== false)
- this._waitForDoubleClick().catch(logError);
- else
- this.menu.toggle();
- }
- return Clutter.EVENT_PROPAGATE;
- }
- vfunc_button_release_event(event) {
- if (!this._indicator.supportsActivation)
- return this._maybeHandleDoubleClick(event);
- return Clutter.EVENT_PROPAGATE;
- }
- vfunc_scroll_event(event) {
- // Since Clutter 1.10, clutter will always send a smooth scrolling event
- // with explicit deltas, no matter what input device is used
- // In fact, for every scroll there will be a smooth and non-smooth scroll
- // event, and we can choose which one we interpret.
- if (event.get_scroll_direction() === Clutter.ScrollDirection.SMOOTH) {
- const [dx, dy] = event.get_scroll_delta();
- this._indicator.scroll(dx, dy);
- return Clutter.EVENT_STOP;
- }
- return Clutter.EVENT_PROPAGATE;
- }
- });
- export const IndicatorStatusTrayIcon = GObject.registerClass(
- class IndicatorTrayIcon extends BaseStatusIcon {
- _init(icon) {
- super._init(0.5, icon.wm_class, icon, {dontCreateMenu: true});
- Util.Logger.debug(`Adding legacy tray icon ${this.uniqueId}`);
- this._box.add_style_class_name('appindicator-trayicons-box');
- this.add_style_class_name('appindicator-icon');
- this.add_style_class_name('tray-icon');
- this.connect('button-press-event', (_actor, _event) => {
- this.add_style_pseudo_class('active');
- return Clutter.EVENT_PROPAGATE;
- });
- this.connect('button-release-event', (_actor, event) => {
- this._icon.click(event);
- this.remove_style_pseudo_class('active');
- return Clutter.EVENT_PROPAGATE;
- });
- this.connect('key-press-event', (_actor, event) => {
- this.add_style_pseudo_class('active');
- this._icon.click(event);
- return Clutter.EVENT_PROPAGATE;
- });
- this.connect('key-release-event', (_actor, event) => {
- this._icon.click(event);
- this.remove_style_pseudo_class('active');
- return Clutter.EVENT_PROPAGATE;
- });
- Util.connectSmart(this._icon, 'destroy', this, () => {
- icon.clear_effects();
- this.destroy();
- });
- const settings = SettingsManager.getDefaultGSettings();
- Util.connectSmart(settings, 'changed::icon-size', this, this._updateIconSize);
- const themeContext = St.ThemeContext.get_for_stage(global.stage);
- Util.connectSmart(themeContext, 'notify::scale-factor', this, () =>
- this._updateIconSize());
- this._updateIconSize();
- }
- _onDestroy() {
- Util.Logger.debug(`Destroying legacy tray icon ${this.uniqueId}`);
- if (this._waitDoubleClickPromise)
- this._waitDoubleClickPromise.cancel();
- super._onDestroy();
- }
- isReady() {
- return !!this._icon;
- }
- get uniqueId() {
- return `legacy:${this._icon.wm_class}:${this._icon.pid}`;
- }
- vfunc_navigate_focus(from, direction) {
- this.grab_key_focus();
- return super.vfunc_navigate_focus(from, direction);
- }
- _getSimulatedButtonEvent(touchEvent) {
- const event = Clutter.Event.new(Clutter.EventType.BUTTON_RELEASE);
- event.set_button(1);
- event.set_time(touchEvent.get_time());
- event.set_flags(touchEvent.get_flags());
- event.set_stage(global.stage);
- event.set_source(touchEvent.get_source());
- event.set_coords(...touchEvent.get_coords());
- event.set_state(touchEvent.get_state());
- return event;
- }
- vfunc_touch_event(event) {
- // Under X11 we rely on emulated pointer events
- if (!imports.gi.Meta.is_wayland_compositor())
- return Clutter.EVENT_PROPAGATE;
- const slot = event.get_event_sequence().get_slot();
- if (!this._touchPressSlot &&
- event.get_type() === Clutter.EventType.TOUCH_BEGIN) {
- this.add_style_pseudo_class('active');
- this._touchButtonEvent = this._getSimulatedButtonEvent(event);
- this._touchPressSlot = slot;
- this._touchDelayPromise = new PromiseUtils.TimeoutPromise(
- AppDisplay.MENU_POPUP_TIMEOUT);
- this._touchDelayPromise.then(() => {
- delete this._touchDelayPromise;
- delete this._touchPressSlot;
- this._touchButtonEvent.set_button(3);
- this._icon.click(this._touchButtonEvent);
- this.remove_style_pseudo_class('active');
- });
- } else if (event.get_type() === Clutter.EventType.TOUCH_END &&
- this._touchPressSlot === slot) {
- delete this._touchPressSlot;
- delete this._touchButtonEvent;
- if (this._touchDelayPromise) {
- this._touchDelayPromise.cancel();
- delete this._touchDelayPromise;
- }
- this._icon.click(this._getSimulatedButtonEvent(event));
- this.remove_style_pseudo_class('active');
- } else if (event.get_type() === Clutter.EventType.TOUCH_UPDATE &&
- this._touchPressSlot === slot) {
- this.add_style_pseudo_class('active');
- this._touchButtonEvent = this._getSimulatedButtonEvent(event);
- }
- return Clutter.EVENT_PROPAGATE;
- }
- vfunc_leave_event(event) {
- this.remove_style_pseudo_class('active');
- if (this._touchDelayPromise) {
- this._touchDelayPromise.cancel();
- delete this._touchDelayPromise;
- }
- return super.vfunc_leave_event(event);
- }
- _updateIconSize() {
- const settings = SettingsManager.getDefaultGSettings();
- const {scaleFactor} = St.ThemeContext.get_for_stage(global.stage);
- let iconSize = settings.get_int('icon-size');
- if (iconSize <= 0)
- iconSize = DEFAULT_ICON_SIZE;
- this.height = -1;
- this._icon.set({
- width: iconSize * scaleFactor,
- height: iconSize * scaleFactor,
- xAlign: Clutter.ActorAlign.CENTER,
- yAlign: Clutter.ActorAlign.CENTER,
- });
- }
- });
|