Blame extensions/apps-menu/extension.js

Packit 8fda33
/* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */
Packit 8fda33
/* exported init enable disable */
Packit 8fda33
Packit 8fda33
const {
Packit 8fda33
    Atk, Clutter, Gio, GLib, GMenu, GObject, Gtk, Meta, Shell, St
Packit 8fda33
} = imports.gi;
Packit 8fda33
const DND = imports.ui.dnd;
Packit 8fda33
const Main = imports.ui.main;
Packit 8fda33
const PanelMenu = imports.ui.panelMenu;
Packit 8fda33
const PopupMenu = imports.ui.popupMenu;
Packit 8fda33
const Signals = imports.signals;
Packit 8fda33
Packit 8fda33
const Gettext = imports.gettext.domain('gnome-shell-extensions');
Packit 8fda33
const _ = Gettext.gettext;
Packit 8fda33
Packit 8fda33
const ExtensionUtils = imports.misc.extensionUtils;
Packit 8fda33
Packit 8fda33
const appSys = Shell.AppSystem.get_default();
Packit 8fda33
Packit 8fda33
const APPLICATION_ICON_SIZE = 32;
Packit 8fda33
const HORIZ_FACTOR = 5;
Packit 8fda33
const MENU_HEIGHT_OFFSET = 132;
Packit 8fda33
const NAVIGATION_REGION_OVERSHOOT = 50;
Packit 8fda33
Packit 8fda33
Gio._promisify(Gio._LocalFilePrototype, 'query_info_async', 'query_info_finish');
Packit 8fda33
Gio._promisify(Gio._LocalFilePrototype, 'set_attributes_async', 'set_attributes_finish');
Packit 8fda33
Packit 8fda33
class ApplicationMenuItem extends PopupMenu.PopupBaseMenuItem {
Packit 8fda33
    constructor(button, app) {
Packit 8fda33
        super();
Packit 8fda33
        this._app = app;
Packit 8fda33
        this._button = button;
Packit 8fda33
Packit 8fda33
        this._iconBin = new St.Bin();
Packit 8fda33
        this.actor.add_child(this._iconBin);
Packit 8fda33
Packit 8fda33
        let appLabel = new St.Label({
Packit 8fda33
            text: app.get_name(),
Packit 8fda33
            y_expand: true,
Packit 8fda33
            y_align: Clutter.ActorAlign.CENTER
Packit 8fda33
        });
Packit 8fda33
        this.actor.add_child(appLabel);
Packit 8fda33
        this.actor.label_actor = appLabel;
Packit 8fda33
Packit 8fda33
        let textureCache = St.TextureCache.get_default();
Packit 8fda33
        let iconThemeChangedId = textureCache.connect('icon-theme-changed',
Packit 8fda33
                                                      this._updateIcon.bind(this));
Packit 8fda33
        this.actor.connect('destroy', () => {
Packit 8fda33
            textureCache.disconnect(iconThemeChangedId);
Packit 8fda33
        });
Packit 8fda33
        this._updateIcon();
Packit 8fda33
Packit 8fda33
        this.actor._delegate = this;
Packit 8fda33
        let draggable = DND.makeDraggable(this.actor);
Packit 8fda33
Packit 8fda33
        let maybeStartDrag = draggable._maybeStartDrag;
Packit 8fda33
        draggable._maybeStartDrag = (event) => {
Packit 8fda33
            if (this._dragEnabled)
Packit 8fda33
                return maybeStartDrag.call(draggable, event);
Packit 8fda33
            return false;
Packit 8fda33
        };
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    activate(event) {
Packit 8fda33
        this._app.open_new_window(-1);
Packit 8fda33
        this._button.selectCategory(null);
Packit 8fda33
        this._button.menu.toggle();
Packit 8fda33
        super.activate(event);
rpm-build 7ca49d
rpm-build 7ca49d
        Main.overview.hide();
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    setActive(active, params) {
Packit 8fda33
        if (active)
Packit 8fda33
            this._button.scrollToButton(this);
Packit 8fda33
        super.setActive(active, params);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    setDragEnabled(enable) {
Packit 8fda33
        this._dragEnabled = enable;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    getDragActor() {
Packit 8fda33
        return this._app.create_icon_texture(APPLICATION_ICON_SIZE);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    getDragActorSource() {
Packit 8fda33
        return this._iconBin;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _updateIcon() {
rpm-build 7ca49d
        let icon = this.getDragActor();
rpm-build 7ca49d
        icon.style_class = 'icon-dropshadow';
rpm-build 7ca49d
        this._iconBin.set_child(icon);
Packit 8fda33
    }
Packit 8fda33
}
Packit 8fda33
Packit 8fda33
class CategoryMenuItem extends PopupMenu.PopupBaseMenuItem {
Packit 8fda33
    constructor(button, category) {
Packit 8fda33
        super();
Packit 8fda33
        this._category = category;
Packit 8fda33
        this._button = button;
Packit 8fda33
Packit 8fda33
        this._oldX = -1;
Packit 8fda33
        this._oldY = -1;
Packit 8fda33
Packit 8fda33
        let name;
Packit 8fda33
        if (this._category)
Packit 8fda33
            name = this._category.get_name();
Packit 8fda33
        else
Packit 8fda33
            name = _('Favorites');
Packit 8fda33
rpm-build b56d08
        let label = new St.Label({ text: name });
rpm-build b56d08
        this.actor.add_child(label);
rpm-build b56d08
        this.actor.label_actor = label;
Packit 8fda33
        this.actor.connect('motion-event', this._onMotionEvent.bind(this));
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    activate(event) {
Packit 8fda33
        this._button.selectCategory(this._category);
Packit 8fda33
        this._button.scrollToCatButton(this);
Packit 8fda33
        super.activate(event);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _isNavigatingSubmenu([x, y]) {
Packit 8fda33
        let [posX, posY] = this.actor.get_transformed_position();
Packit 8fda33
Packit 8fda33
        if (this._oldX == -1) {
Packit 8fda33
            this._oldX = x;
Packit 8fda33
            this._oldY = y;
Packit 8fda33
            return true;
Packit 8fda33
        }
Packit 8fda33
Packit 8fda33
        let deltaX = Math.abs(x - this._oldX);
Packit 8fda33
        let deltaY = Math.abs(y - this._oldY);
Packit 8fda33
Packit 8fda33
        this._oldX = x;
Packit 8fda33
        this._oldY = y;
Packit 8fda33
Packit 8fda33
        // If it lies outside the x-coordinates then it is definitely outside.
Packit 8fda33
        if (posX > x || posX + this.actor.width < x)
Packit 8fda33
            return false;
Packit 8fda33
Packit 8fda33
        // If it lies inside the menu item then it is definitely inside.
Packit 8fda33
        if (posY <= y && posY + this.actor.height >= y)
Packit 8fda33
            return true;
Packit 8fda33
Packit 8fda33
        // We want the keep-up triangle only if the movement is more
Packit 8fda33
        // horizontal than vertical.
Packit 8fda33
        if (deltaX * HORIZ_FACTOR < deltaY)
Packit 8fda33
            return false;
Packit 8fda33
Packit 8fda33
        // Check whether the point lies inside triangle ABC, and a similar
Packit 8fda33
        // triangle on the other side of the menu item.
Packit 8fda33
        //
Packit 8fda33
        //   +---------------------+
Packit 8fda33
        //   | menu item           |
Packit 8fda33
        // A +---------------------+ C
Packit 8fda33
        //              P          |
Packit 8fda33
        //                         B
Packit 8fda33
Packit 8fda33
        // Ensure that the point P always lies below line AC so that we can
Packit 8fda33
        // only check for triangle ABC.
Packit 8fda33
        if (posY > y) {
Packit 8fda33
            let offset = posY - y;
Packit 8fda33
            y = posY + this.actor.height + offset;
Packit 8fda33
        }
Packit 8fda33
Packit 8fda33
        // Ensure that A is (0, 0).
Packit 8fda33
        x -= posX;
Packit 8fda33
        y -= posY + this.actor.height;
Packit 8fda33
Packit 8fda33
        // Check which side of line AB the point P lies on by taking the
Packit 8fda33
        // cross-product of AB and AP. See:
Packit 8fda33
        // http://stackoverflow.com/questions/3461453/determine-which-side-of-a-line-a-point-lies
Packit 8fda33
        if (((this.actor.width * y) - (NAVIGATION_REGION_OVERSHOOT * x)) <= 0)
Packit 8fda33
            return true;
Packit 8fda33
Packit 8fda33
        return false;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onMotionEvent(actor, event) {
Packit 8fda33
        if (!Clutter.get_pointer_grab()) {
Packit 8fda33
            this._oldX = -1;
Packit 8fda33
            this._oldY = -1;
Packit 8fda33
            Clutter.grab_pointer(this.actor);
Packit 8fda33
        }
Packit 8fda33
        this.actor.hover = true;
Packit 8fda33
Packit 8fda33
        if (this._isNavigatingSubmenu(event.get_coords()))
Packit 8fda33
            return true;
Packit 8fda33
Packit 8fda33
        this._oldX = -1;
Packit 8fda33
        this._oldY = -1;
Packit 8fda33
        this.actor.hover = false;
Packit 8fda33
        Clutter.ungrab_pointer();
Packit 8fda33
Packit 8fda33
        let source = event.get_source();
Packit 8fda33
        if (source instanceof St.Widget)
Packit 8fda33
            source.sync_hover();
Packit 8fda33
Packit 8fda33
        return false;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    setActive(active, params) {
Packit 8fda33
        if (active) {
Packit 8fda33
            this._button.selectCategory(this._category);
Packit 8fda33
            this._button.scrollToCatButton(this);
Packit 8fda33
        }
Packit 8fda33
        super.setActive(active, params);
Packit 8fda33
    }
Packit 8fda33
}
Packit 8fda33
Packit 8fda33
class ApplicationsMenu extends PopupMenu.PopupMenu {
Packit 8fda33
    constructor(sourceActor, arrowAlignment, arrowSide, button) {
Packit 8fda33
        super(sourceActor, arrowAlignment, arrowSide);
Packit 8fda33
        this._button = button;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    isEmpty() {
Packit 8fda33
        return false;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    toggle() {
rpm-build 7ca49d
        if (this.isOpen)
Packit 8fda33
            this._button.selectCategory(null);
Packit 8fda33
        super.toggle();
Packit 8fda33
    }
Packit 8fda33
}
Packit 8fda33
Packit 8fda33
class DesktopTarget {
Packit 8fda33
    constructor() {
Packit 8fda33
        this._desktop = null;
Packit 8fda33
        this._desktopDestroyedId = 0;
Packit 8fda33
Packit 8fda33
        this._windowAddedId =
Packit 8fda33
            global.window_group.connect('actor-added',
Packit 8fda33
                                        this._onWindowAdded.bind(this));
Packit 8fda33
Packit 8fda33
        global.get_window_actors().forEach(a => {
Packit 8fda33
            this._onWindowAdded(a.get_parent(), a);
Packit 8fda33
        });
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    get hasDesktop() {
Packit 8fda33
        return this._desktop != null;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onWindowAdded(group, actor) {
Packit 8fda33
        if (!(actor instanceof Meta.WindowActor))
Packit 8fda33
            return;
Packit 8fda33
Packit 8fda33
        if (actor.meta_window.get_window_type() == Meta.WindowType.DESKTOP)
Packit 8fda33
            this._setDesktop(actor);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _setDesktop(desktop) {
Packit 8fda33
        if (this._desktop) {
Packit 8fda33
            this._desktop.disconnect(this._desktopDestroyedId);
Packit 8fda33
            this._desktopDestroyedId = 0;
Packit 8fda33
Packit 8fda33
            delete this._desktop._delegate;
Packit 8fda33
        }
Packit 8fda33
Packit 8fda33
        this._desktop = desktop;
Packit 8fda33
        this.emit('desktop-changed');
Packit 8fda33
Packit 8fda33
        if (this._desktop) {
Packit 8fda33
            this._desktopDestroyedId = this._desktop.connect('destroy', () => {
Packit 8fda33
                this._setDesktop(null);
Packit 8fda33
            });
Packit 8fda33
            this._desktop._delegate = this;
Packit 8fda33
        }
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _getSourceAppInfo(source) {
Packit 8fda33
        if (!(source instanceof ApplicationMenuItem))
Packit 8fda33
            return null;
Packit 8fda33
        return source._app.app_info;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    async _markTrusted(file) {
Packit 8fda33
        let modeAttr = Gio.FILE_ATTRIBUTE_UNIX_MODE;
Packit 8fda33
        let trustedAttr = 'metadata::trusted';
Packit 8fda33
        let queryFlags = Gio.FileQueryInfoFlags.NONE;
Packit 8fda33
        let ioPriority = GLib.PRIORITY_DEFAULT;
Packit 8fda33
Packit 8fda33
        try {
Packit 8fda33
            let info = await file.query_info_async(modeAttr, queryFlags, ioPriority, null);
Packit 8fda33
Packit 8fda33
            let mode = info.get_attribute_uint32(modeAttr) | 0o100;
Packit 8fda33
            info.set_attribute_uint32(modeAttr, mode);
Packit 8fda33
            info.set_attribute_string(trustedAttr, 'yes');
Packit 8fda33
            await file.set_attributes_async(info, queryFlags, ioPriority, null);
Packit 8fda33
Packit 8fda33
            // Hack: force nautilus to reload file info
Packit 8fda33
            info = new Gio.FileInfo();
Packit 8fda33
            info.set_attribute_uint64(Gio.FILE_ATTRIBUTE_TIME_ACCESS,
Packit 8fda33
                                      GLib.get_real_time());
Packit 8fda33
            try {
Packit 8fda33
                await file.set_attributes_async(info, queryFlags, ioPriority, null);
Packit 8fda33
            } catch (e) {
Packit 8fda33
                log(`Failed to update access time: ${e.message}`);
Packit 8fda33
            }
Packit 8fda33
        } catch (e) {
Packit 8fda33
            log(`Failed to mark file as trusted: ${e.message}`);
Packit 8fda33
        }
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    destroy() {
Packit 8fda33
        if (this._windowAddedId)
Packit 8fda33
            global.window_group.disconnect(this._windowAddedId);
Packit 8fda33
        this._windowAddedId = 0;
Packit 8fda33
Packit 8fda33
        this._setDesktop(null);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    handleDragOver(source, _actor, _x, _y, _time) {
Packit 8fda33
        let appInfo = this._getSourceAppInfo(source);
Packit 8fda33
        if (!appInfo)
Packit 8fda33
            return DND.DragMotionResult.CONTINUE;
Packit 8fda33
Packit 8fda33
        return DND.DragMotionResult.COPY_DROP;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    acceptDrop(source, _actor, _x, _y, _time) {
Packit 8fda33
        let appInfo = this._getSourceAppInfo(source);
Packit 8fda33
        if (!appInfo)
Packit 8fda33
            return false;
Packit 8fda33
Packit 8fda33
        this.emit('app-dropped');
Packit 8fda33
Packit 8fda33
        let desktop = GLib.get_user_special_dir(GLib.UserDirectory.DIRECTORY_DESKTOP);
Packit 8fda33
Packit 8fda33
        let src = Gio.File.new_for_path(appInfo.get_filename());
Packit 8fda33
        let dst = Gio.File.new_for_path(GLib.build_filenamev([desktop, src.get_basename()]));
Packit 8fda33
Packit 8fda33
        try {
Packit 8fda33
            // copy_async() isn't introspectable :-(
Packit 8fda33
            src.copy(dst, Gio.FileCopyFlags.OVERWRITE, null, null);
Packit 8fda33
            this._markTrusted(dst);
Packit 8fda33
        } catch (e) {
Packit 8fda33
            log(`Failed to copy to desktop: ${e.message}`);
Packit 8fda33
        }
Packit 8fda33
Packit 8fda33
        return true;
Packit 8fda33
    }
Packit 8fda33
}
Packit 8fda33
Signals.addSignalMethods(DesktopTarget.prototype);
Packit 8fda33
Packit 8fda33
let ApplicationsButton = GObject.registerClass(
Packit 8fda33
class ApplicationsButton extends PanelMenu.Button {
rpm-build 7ca49d
    _init(includeIcon) {
Packit 8fda33
        super._init(1.0, null, false);
Packit 8fda33
Packit 8fda33
        this.setMenu(new ApplicationsMenu(this, 1.0, St.Side.TOP, this));
Packit 8fda33
        Main.panel.menuManager.addMenu(this.menu);
Packit 8fda33
Packit 8fda33
        // At this moment applications menu is not keyboard navigable at
Packit 8fda33
        // all (so not accessible), so it doesn't make sense to set as
Packit 8fda33
        // role ATK_ROLE_MENU like other elements of the panel.
Packit 8fda33
        this.accessible_role = Atk.Role.LABEL;
Packit 8fda33
Packit 8fda33
        let hbox = new St.BoxLayout({ style_class: 'panel-status-menu-box' });
Packit 8fda33
rpm-build 7ac357
        let iconFile = Gio.File.new_for_path(
rpm-build 7ac357
            '/usr/share/icons/hicolor/scalable/apps/start-here.svg');
rpm-build 7ac357
        this._icon = new St.Icon({
rpm-build 7ac357
            gicon: new Gio.FileIcon({ file: iconFile }),
rpm-build 7ca49d
            style_class: 'panel-logo-icon',
rpm-build 7ca49d
            visible: includeIcon
rpm-build 7ac357
        });
rpm-build 7ac357
        hbox.add_actor(this._icon);
rpm-build 7ac357
Packit 8fda33
        this._label = new St.Label({
Packit 8fda33
            text: _('Applications'),
Packit 8fda33
            y_expand: true,
Packit 8fda33
            y_align: Clutter.ActorAlign.CENTER
Packit 8fda33
        });
Packit 8fda33
        hbox.add_child(this._label);
Packit 8fda33
        hbox.add_child(PopupMenu.arrowIcon(St.Side.BOTTOM));
Packit 8fda33
Packit 8fda33
        this.add_actor(hbox);
Packit 8fda33
        this.name = 'panelApplications';
Packit 8fda33
        this.label_actor = this._label;
Packit 8fda33
Packit 8fda33
        this._showingId = Main.overview.connect('showing', () => {
Packit 8fda33
            this.add_accessible_state (Atk.StateType.CHECKED);
Packit 8fda33
        });
Packit 8fda33
        this._hidingId = Main.overview.connect('hiding', () => {
Packit 8fda33
            this.remove_accessible_state (Atk.StateType.CHECKED);
Packit 8fda33
        });
Packit 8fda33
        Main.layoutManager.connect('startup-complete',
Packit 8fda33
                                   this._setKeybinding.bind(this));
Packit 8fda33
        this._setKeybinding();
Packit 8fda33
Packit 8fda33
        this._desktopTarget = new DesktopTarget();
Packit 8fda33
        this._desktopTarget.connect('app-dropped', () => {
Packit 8fda33
            this.menu.close();
Packit 8fda33
        });
Packit 8fda33
        this._desktopTarget.connect('desktop-changed', () => {
Packit 8fda33
            this._applicationsButtons.forEach(item => {
Packit 8fda33
                item.setDragEnabled(this._desktopTarget.hasDesktop);
Packit 8fda33
            });
Packit 8fda33
        });
Packit 8fda33
Packit 8fda33
        this._tree = new GMenu.Tree({ menu_basename: 'applications.menu' });
Packit 8fda33
        this._treeChangedId = this._tree.connect('changed',
Packit 8fda33
                                                 this._onTreeChanged.bind(this));
Packit 8fda33
Packit 8fda33
        this._applicationsButtons = new Map();
Packit 8fda33
        this.reloadFlag = false;
Packit 8fda33
        this._createLayout();
Packit 8fda33
        this._display();
Packit 8fda33
        this._installedChangedId = appSys.connect('installed-changed',
Packit 8fda33
                                                  this._onTreeChanged.bind(this));
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onTreeChanged() {
Packit 8fda33
        if (this.menu.isOpen) {
Packit 8fda33
            this._redisplay();
Packit 8fda33
            this.mainBox.show();
Packit 8fda33
        } else {
Packit 8fda33
            this.reloadFlag = true;
Packit 8fda33
        }
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _createVertSeparator() {
Packit 8fda33
        let separator = new St.DrawingArea({
Packit 8fda33
            style_class: 'calendar-vertical-separator',
Packit 8fda33
            pseudo_class: 'highlighted'
Packit 8fda33
        });
Packit 8fda33
        separator.connect('repaint', this._onVertSepRepaint.bind(this));
Packit 8fda33
        return separator;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onDestroy() {
rpm-build 007d01
        super._onDestroy();
rpm-build 007d01
Packit 8fda33
        Main.overview.disconnect(this._showingId);
Packit 8fda33
        Main.overview.disconnect(this._hidingId);
Packit 8fda33
        appSys.disconnect(this._installedChangedId);
Packit 8fda33
        this._tree.disconnect(this._treeChangedId);
Packit 8fda33
        this._tree = null;
Packit 8fda33
Packit 8fda33
        let handler = Main.sessionMode.hasOverview ?
Packit 8fda33
            Main.overview.toggle.bind(Main.overview) : null;
Packit 8fda33
        Main.wm.setCustomKeybindingHandler('panel-main-menu',
Packit 8fda33
                                           Shell.ActionMode.NORMAL |
Packit 8fda33
                                           Shell.ActionMode.OVERVIEW,
Packit 8fda33
                                           handler);
Packit 8fda33
Packit 8fda33
        this._desktopTarget.destroy();
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onMenuKeyPress(actor, event) {
Packit 8fda33
        let symbol = event.get_key_symbol();
Packit 8fda33
        if (symbol == Clutter.KEY_Left || symbol == Clutter.KEY_Right) {
Packit 8fda33
            let direction = symbol == Clutter.KEY_Left ?
Packit 8fda33
                Gtk.DirectionType.LEFT : Gtk.DirectionType.RIGHT;
Packit 8fda33
            if (this.menu.actor.navigate_focus(global.stage.key_focus, direction, false))
Packit 8fda33
                return true;
Packit 8fda33
        }
Packit 8fda33
        return super._onMenuKeyPress(actor, event);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onVertSepRepaint(area) {
Packit 8fda33
        let cr = area.get_context();
Packit 8fda33
        let themeNode = area.get_theme_node();
Packit 8fda33
        let [width, height] = area.get_surface_size();
Packit 8fda33
        let stippleColor = themeNode.get_color('-stipple-color');
Packit 8fda33
        let stippleWidth = themeNode.get_length('-stipple-width');
Packit 8fda33
        let x = Math.floor(width / 2) + 0.5;
Packit 8fda33
        cr.moveTo(x, 0);
Packit 8fda33
        cr.lineTo(x, height);
Packit 8fda33
        Clutter.cairo_set_source_color(cr, stippleColor);
Packit 8fda33
        cr.setDash([1, 3], 1); // Hard-code for now
Packit 8fda33
        cr.setLineWidth(stippleWidth);
Packit 8fda33
        cr.stroke();
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _onOpenStateChanged(menu, open) {
Packit 8fda33
        if (open) {
Packit 8fda33
            if (this.reloadFlag) {
Packit 8fda33
                this._redisplay();
Packit 8fda33
                this.reloadFlag = false;
Packit 8fda33
            }
Packit 8fda33
            this.mainBox.show();
Packit 8fda33
        }
Packit 8fda33
        super._onOpenStateChanged(menu, open);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _setKeybinding() {
Packit 8fda33
        Main.wm.setCustomKeybindingHandler('panel-main-menu',
Packit 8fda33
                                           Shell.ActionMode.NORMAL |
Packit 8fda33
                                           Shell.ActionMode.OVERVIEW,
Packit 8fda33
                                           () => this.menu.toggle());
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _redisplay() {
Packit 8fda33
        this.applicationsBox.destroy_all_children();
Packit 8fda33
        this.categoriesBox.destroy_all_children();
Packit 8fda33
        this._display();
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _loadCategory(categoryId, dir) {
Packit 8fda33
        let iter = dir.iter();
Packit 8fda33
        let nextType;
Packit 8fda33
        while ((nextType = iter.next()) != GMenu.TreeItemType.INVALID) {
Packit 8fda33
            if (nextType == GMenu.TreeItemType.ENTRY) {
Packit 8fda33
                let entry = iter.get_entry();
Packit 8fda33
                let id;
Packit 8fda33
                try {
Packit 8fda33
                    id = entry.get_desktop_file_id(); // catch non-UTF8 filenames
Packit 8fda33
                } catch (e) {
Packit 8fda33
                    continue;
Packit 8fda33
                }
Packit 8fda33
                let app = appSys.lookup_app(id);
Packit 8fda33
                if (!app)
Packit 8fda33
                    app = new Shell.App({ app_info: entry.get_app_info() });
Packit 8fda33
                if (app.get_app_info().should_show())
Packit 8fda33
                    this.applicationsByCategory[categoryId].push(app);
Packit 8fda33
            } else if (nextType == GMenu.TreeItemType.SEPARATOR) {
Packit 8fda33
                this.applicationsByCategory[categoryId].push('separator');
Packit 8fda33
            } else if (nextType == GMenu.TreeItemType.DIRECTORY) {
Packit 8fda33
                let subdir = iter.get_directory();
Packit 8fda33
                if (!subdir.get_is_nodisplay())
Packit 8fda33
                    this._loadCategory(categoryId, subdir);
Packit 8fda33
            }
Packit 8fda33
        }
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    scrollToButton(button) {
Packit 8fda33
        let appsScrollBoxAdj = this.applicationsScrollBox.get_vscroll_bar().get_adjustment();
Packit 8fda33
        let appsScrollBoxAlloc = this.applicationsScrollBox.get_allocation_box();
Packit 8fda33
        let currentScrollValue = appsScrollBoxAdj.get_value();
Packit 8fda33
        let boxHeight = appsScrollBoxAlloc.y2 - appsScrollBoxAlloc.y1;
Packit 8fda33
        let buttonAlloc = button.actor.get_allocation_box();
Packit 8fda33
        let newScrollValue = currentScrollValue;
Packit 8fda33
        if (currentScrollValue > buttonAlloc.y1 - 10)
Packit 8fda33
            newScrollValue = buttonAlloc.y1 - 10;
Packit 8fda33
        if (boxHeight + currentScrollValue < buttonAlloc.y2 + 10)
Packit 8fda33
            newScrollValue = buttonAlloc.y2 - boxHeight + 10;
Packit 8fda33
        if (newScrollValue != currentScrollValue)
Packit 8fda33
            appsScrollBoxAdj.set_value(newScrollValue);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    scrollToCatButton(button) {
Packit 8fda33
        let catsScrollBoxAdj = this.categoriesScrollBox.get_vscroll_bar().get_adjustment();
Packit 8fda33
        let catsScrollBoxAlloc = this.categoriesScrollBox.get_allocation_box();
Packit 8fda33
        let currentScrollValue = catsScrollBoxAdj.get_value();
Packit 8fda33
        let boxHeight = catsScrollBoxAlloc.y2 - catsScrollBoxAlloc.y1;
Packit 8fda33
        let buttonAlloc = button.actor.get_allocation_box();
Packit 8fda33
        let newScrollValue = currentScrollValue;
Packit 8fda33
        if (currentScrollValue > buttonAlloc.y1 - 10)
Packit 8fda33
            newScrollValue = buttonAlloc.y1 - 10;
Packit 8fda33
        if (boxHeight + currentScrollValue < buttonAlloc.y2 + 10)
Packit 8fda33
            newScrollValue = buttonAlloc.y2 - boxHeight + 10;
Packit 8fda33
        if (newScrollValue != currentScrollValue)
Packit 8fda33
            catsScrollBoxAdj.set_value(newScrollValue);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _createLayout() {
Packit 8fda33
        let section = new PopupMenu.PopupMenuSection();
Packit 8fda33
        this.menu.addMenuItem(section);
Packit 8fda33
        this.mainBox = new St.BoxLayout({ vertical: false });
Packit 8fda33
        this.leftBox = new St.BoxLayout({ vertical: true });
Packit 8fda33
        this.applicationsScrollBox = new St.ScrollView({
Packit 8fda33
            x_fill: true,
Packit 8fda33
            y_fill: false,
Packit 8fda33
            y_align: St.Align.START,
Packit 8fda33
            style_class: 'apps-menu vfade'
Packit 8fda33
        });
Packit 8fda33
        this.applicationsScrollBox.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC);
Packit 8fda33
        let vscroll = this.applicationsScrollBox.get_vscroll_bar();
Packit 8fda33
        vscroll.connect('scroll-start', () => {
Packit 8fda33
            this.menu.passEvents = true;
Packit 8fda33
        });
Packit 8fda33
        vscroll.connect('scroll-stop', () => {
Packit 8fda33
            this.menu.passEvents = false;
Packit 8fda33
        });
Packit 8fda33
        this.categoriesScrollBox = new St.ScrollView({
Packit 8fda33
            x_fill: true,
Packit 8fda33
            y_fill: false,
Packit 8fda33
            y_align: St.Align.START,
Packit 8fda33
            style_class: 'vfade'
Packit 8fda33
        });
Packit 8fda33
        this.categoriesScrollBox.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC);
Packit 8fda33
        vscroll = this.categoriesScrollBox.get_vscroll_bar();
Packit 8fda33
        vscroll.connect('scroll-start', () => this.menu.passEvents = true);
Packit 8fda33
        vscroll.connect('scroll-stop', () => this.menu.passEvents = false);
Packit 8fda33
        this.leftBox.add(this.categoriesScrollBox, {
Packit 8fda33
            expand: true,
Packit 8fda33
            x_fill: true,
Packit 8fda33
            y_fill: true,
Packit 8fda33
            y_align: St.Align.START
Packit 8fda33
        });
Packit 8fda33
Packit 8fda33
        this.applicationsBox = new St.BoxLayout({ vertical: true });
Packit 8fda33
        this.applicationsScrollBox.add_actor(this.applicationsBox);
Packit 8fda33
        this.categoriesBox = new St.BoxLayout({ vertical: true });
Packit 8fda33
        this.categoriesScrollBox.add_actor(this.categoriesBox);
Packit 8fda33
Packit 8fda33
        this.mainBox.add(this.leftBox);
Packit 8fda33
        this.mainBox.add(this._createVertSeparator(), {
Packit 8fda33
            expand: false,
Packit 8fda33
            x_fill: false,
Packit 8fda33
            y_fill: true
Packit 8fda33
        });
Packit 8fda33
        this.mainBox.add(this.applicationsScrollBox, {
Packit 8fda33
            expand: true,
Packit 8fda33
            x_fill: true,
Packit 8fda33
            y_fill: true
Packit 8fda33
        });
Packit 8fda33
        section.actor.add_actor(this.mainBox);
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _display() {
Packit 8fda33
        this._applicationsButtons.clear();
Packit 8fda33
        this.mainBox.style = 'width: 35em;';
Packit 8fda33
        this.mainBox.hide();
Packit 8fda33
Packit 8fda33
        //Load categories
Packit 8fda33
        this.applicationsByCategory = {};
Packit 8fda33
        this._tree.load_sync();
Packit 8fda33
        let root = this._tree.get_root_directory();
Packit 8fda33
        let categoryMenuItem = new CategoryMenuItem(this, null);
Packit 8fda33
        this.categoriesBox.add_actor(categoryMenuItem.actor);
Packit 8fda33
        let iter = root.iter();
Packit 8fda33
        let nextType;
Packit 8fda33
        while ((nextType = iter.next()) != GMenu.TreeItemType.INVALID) {
Packit 8fda33
            if (nextType != GMenu.TreeItemType.DIRECTORY)
Packit 8fda33
                continue;
Packit 8fda33
Packit 8fda33
            let dir = iter.get_directory();
Packit 8fda33
            if (dir.get_is_nodisplay())
Packit 8fda33
                continue;
Packit 8fda33
Packit 8fda33
            let categoryId = dir.get_menu_id();
Packit 8fda33
            this.applicationsByCategory[categoryId] = [];
Packit 8fda33
            this._loadCategory(categoryId, dir);
Packit 8fda33
            if (this.applicationsByCategory[categoryId].length > 0) {
Packit 8fda33
                let categoryMenuItem = new CategoryMenuItem(this, dir);
Packit 8fda33
                this.categoriesBox.add_actor(categoryMenuItem.actor);
Packit 8fda33
            }
Packit 8fda33
        }
Packit 8fda33
Packit 8fda33
        //Load applications
Packit 8fda33
        this._displayButtons(this._listApplications(null));
Packit 8fda33
Packit 8fda33
        let themeContext = St.ThemeContext.get_for_stage(global.stage);
Packit 8fda33
        let scaleFactor = themeContext.scale_factor;
Packit 8fda33
        let categoriesHeight = this.categoriesBox.height / scaleFactor;
Packit 8fda33
        let height = Math.round(categoriesHeight) + MENU_HEIGHT_OFFSET;
Packit 8fda33
        this.mainBox.style += `height: ${height}px`;
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    selectCategory(dir) {
Packit 8fda33
        this.applicationsBox.get_children().forEach(c => {
Packit 8fda33
            if (c._delegate instanceof PopupMenu.PopupSeparatorMenuItem)
Packit 8fda33
                c._delegate.destroy();
Packit 8fda33
            else
Packit 8fda33
                this.applicationsBox.remove_actor(c);
Packit 8fda33
        });
Packit 8fda33
Packit 8fda33
        if (dir)
Packit 8fda33
            this._displayButtons(this._listApplications(dir.get_menu_id()));
Packit 8fda33
        else
Packit 8fda33
            this._displayButtons(this._listApplications(null));
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _displayButtons(apps) {
Packit 8fda33
        for (let i = 0; i < apps.length; i++) {
Packit 8fda33
            let app = apps[i];
Packit 8fda33
            let item;
Packit 8fda33
            if (app instanceof Shell.App)
Packit 8fda33
                item = this._applicationsButtons.get(app);
Packit 8fda33
            else
Packit 8fda33
                item = new PopupMenu.PopupSeparatorMenuItem();
Packit 8fda33
            if (!item) {
Packit 8fda33
                item = new ApplicationMenuItem(this, app);
Packit 8fda33
                item.setDragEnabled(this._desktopTarget.hasDesktop);
Packit 8fda33
                this._applicationsButtons.set(app, item);
Packit 8fda33
            }
Packit 8fda33
            if (!item.actor.get_parent())
Packit 8fda33
                this.applicationsBox.add_actor(item.actor);
Packit 8fda33
        }
Packit 8fda33
    }
Packit 8fda33
Packit 8fda33
    _listApplications(categoryMenuId) {
Packit 8fda33
        let applist;
Packit 8fda33
Packit 8fda33
        if (categoryMenuId) {
Packit 8fda33
            applist = this.applicationsByCategory[categoryMenuId];
Packit 8fda33
        } else {
Packit 8fda33
            applist = new Array();
Packit 8fda33
            let favorites = global.settings.get_strv('favorite-apps');
Packit 8fda33
            for (let i = 0; i < favorites.length; i++) {
Packit 8fda33
                let app = appSys.lookup_app(favorites[i]);
Packit 8fda33
                if (app)
Packit 8fda33
                    applist.push(app);
Packit 8fda33
            }
Packit 8fda33
        }
Packit 8fda33
Packit 8fda33
        return applist;
Packit 8fda33
    }
Packit 8fda33
});
Packit 8fda33
Packit 8fda33
let appsMenuButton;
Packit 8fda33
Packit 8fda33
function enable() {
rpm-build 7ca49d
    let index = Main.sessionMode.panel.left.indexOf('activities') + 1;
rpm-build 7ca49d
    appsMenuButton = new ApplicationsButton(index == 0);
rpm-build 7ca49d
    Main.panel.addToStatusArea('apps-menu', appsMenuButton, index, 'left');
Packit 8fda33
}
Packit 8fda33
Packit 8fda33
function disable() {
Packit 8fda33
    Main.panel.menuManager.removeMenu(appsMenuButton.menu);
Packit 8fda33
    appsMenuButton.destroy();
Packit 8fda33
}
Packit 8fda33
Packit 8fda33
function init() {
Packit 8fda33
    ExtensionUtils.initTranslations();
Packit 8fda33
}