2011-09-28 09:16:26 -04:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2019-01-31 09:07:06 -05:00
|
|
|
/* exported Dash */
|
Rewrite Dash, remove hardcoded width/height from GenericDisplay
This patch is a near-total rewrite of the Dash. First, the dash
code moves into a separate file, dash.js.
Inside dash.js, the components are more broken up into separate
classes; in particular there's now a Pane class and a MoreLink
class. Instead of each section of the dash, when activated,
attempting to close all N-1 other sections, instead there
is the concept of a single "active pane", and when e.g. activating
the More link for documents, if we know there's an active pane
which happens to be the apps, close it.
Many redundant containers were removed from the dash, and all
manual width, height and x/y offsets are entirely gone. We move
the visual apperance closer to the design by using the view-more.svg,
etc.
To complete the removal of height/width calculations from the dash,
we also had to do the same for GenericDisplay. Also clean up
the positioning inside overlay.js so calculation of children's
positioning is inside a single function that flows from screen.width
and screen.height, so in the future we can stop passing the width
into the Dash constructor and call this once and work on screen
resizing.
2009-07-31 22:12:01 -04:00
|
|
|
|
2019-02-20 14:54:29 -05:00
|
|
|
const { Clutter, GLib, GObject,
|
|
|
|
Graphene, Meta, Shell, St } = imports.gi;
|
Rewrite Dash, remove hardcoded width/height from GenericDisplay
This patch is a near-total rewrite of the Dash. First, the dash
code moves into a separate file, dash.js.
Inside dash.js, the components are more broken up into separate
classes; in particular there's now a Pane class and a MoreLink
class. Instead of each section of the dash, when activated,
attempting to close all N-1 other sections, instead there
is the concept of a single "active pane", and when e.g. activating
the More link for documents, if we know there's an active pane
which happens to be the apps, close it.
Many redundant containers were removed from the dash, and all
manual width, height and x/y offsets are entirely gone. We move
the visual apperance closer to the design by using the view-more.svg,
etc.
To complete the removal of height/width calculations from the dash,
we also had to do the same for GenericDisplay. Also clean up
the positioning inside overlay.js so calculation of children's
positioning is inside a single function that flows from screen.width
and screen.height, so in the future we can stop passing the width
into the Dash constructor and call this once and work on screen
resizing.
2009-07-31 22:12:01 -04:00
|
|
|
|
|
|
|
const AppDisplay = imports.ui.appDisplay;
|
2010-07-21 19:29:02 -04:00
|
|
|
const AppFavorites = imports.ui.appFavorites;
|
2010-02-07 12:59:30 -05:00
|
|
|
const DND = imports.ui.dnd;
|
2010-11-07 20:51:02 -05:00
|
|
|
const IconGrid = imports.ui.iconGrid;
|
Rewrite Dash, remove hardcoded width/height from GenericDisplay
This patch is a near-total rewrite of the Dash. First, the dash
code moves into a separate file, dash.js.
Inside dash.js, the components are more broken up into separate
classes; in particular there's now a Pane class and a MoreLink
class. Instead of each section of the dash, when activated,
attempting to close all N-1 other sections, instead there
is the concept of a single "active pane", and when e.g. activating
the More link for documents, if we know there's an active pane
which happens to be the apps, close it.
Many redundant containers were removed from the dash, and all
manual width, height and x/y offsets are entirely gone. We move
the visual apperance closer to the design by using the view-more.svg,
etc.
To complete the removal of height/width calculations from the dash,
we also had to do the same for GenericDisplay. Also clean up
the positioning inside overlay.js so calculation of children's
positioning is inside a single function that flows from screen.width
and screen.height, so in the future we can stop passing the width
into the Dash constructor and call this once and work on screen
resizing.
2009-07-31 22:12:01 -04:00
|
|
|
const Main = imports.ui.main;
|
2009-11-29 17:45:30 -05:00
|
|
|
|
2019-08-01 19:13:10 -04:00
|
|
|
var DASH_ANIMATION_TIME = 200;
|
|
|
|
var DASH_ITEM_LABEL_SHOW_TIME = 150;
|
|
|
|
var DASH_ITEM_LABEL_HIDE_TIME = 100;
|
2017-07-18 13:47:27 -04:00
|
|
|
var DASH_ITEM_HOVER_TIMEOUT = 300;
|
2011-01-21 06:06:13 -05:00
|
|
|
|
2012-10-02 20:13:47 -04:00
|
|
|
function getAppFromSource(source) {
|
2019-08-19 20:51:42 -04:00
|
|
|
if (source instanceof AppDisplay.AppIcon)
|
2012-11-03 16:14:53 -04:00
|
|
|
return source.app;
|
2019-08-19 20:51:42 -04:00
|
|
|
else
|
2012-10-02 20:13:47 -04:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-07-16 05:24:13 -04:00
|
|
|
var DashIcon = GObject.registerClass(
|
|
|
|
class DashIcon extends AppDisplay.AppIcon {
|
|
|
|
_init(app) {
|
|
|
|
super._init(app, {
|
2019-07-12 18:32:54 -04:00
|
|
|
setSizeManually: true,
|
2019-08-20 17:43:54 -04:00
|
|
|
showLabel: false,
|
2019-07-12 18:32:54 -04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-01-07 08:05:18 -05:00
|
|
|
popupMenu() {
|
|
|
|
super.popupMenu(St.Side.BOTTOM);
|
|
|
|
}
|
|
|
|
|
2020-10-02 11:10:42 -04:00
|
|
|
// Disable scale-n-fade methods used during DND by parent
|
|
|
|
scaleAndFade() {
|
2019-07-12 18:32:54 -04:00
|
|
|
}
|
|
|
|
|
2020-10-02 11:10:42 -04:00
|
|
|
undoScaleAndFade() {
|
2019-07-12 18:32:54 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
handleDragOver() {
|
|
|
|
return DND.DragMotionResult.CONTINUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
acceptDrop() {
|
|
|
|
return false;
|
|
|
|
}
|
2019-07-16 05:24:13 -04:00
|
|
|
});
|
2019-07-12 18:32:54 -04:00
|
|
|
|
2011-01-20 10:23:43 -05:00
|
|
|
// A container like StBin, but taking the child's scale into account
|
|
|
|
// when requesting a size
|
2017-10-30 21:23:39 -04:00
|
|
|
var DashItemContainer = GObject.registerClass(
|
|
|
|
class DashItemContainer extends St.Widget {
|
2017-10-30 20:03:21 -04:00
|
|
|
_init() {
|
2017-10-30 21:23:39 -04:00
|
|
|
super._init({ style_class: 'dash-item-container',
|
2019-02-20 14:54:29 -05:00
|
|
|
pivot_point: new Graphene.Point({ x: .5, y: .5 }),
|
2018-07-21 01:08:33 -04:00
|
|
|
scale_x: 0,
|
|
|
|
scale_y: 0,
|
|
|
|
opacity: 0,
|
2018-09-13 16:37:54 -04:00
|
|
|
x_expand: true,
|
|
|
|
x_align: Clutter.ActorAlign.CENTER });
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2012-10-03 01:22:46 -04:00
|
|
|
this._labelText = "";
|
2019-01-28 20:27:05 -05:00
|
|
|
this.label = new St.Label({ style_class: 'dash-label' });
|
2012-10-03 01:19:36 -04:00
|
|
|
this.label.hide();
|
|
|
|
Main.layoutManager.addChrome(this.label);
|
2012-12-22 08:57:06 -05:00
|
|
|
this.label_actor = this.label;
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2011-01-20 10:23:43 -05:00
|
|
|
this.child = null;
|
2011-09-19 17:47:16 -04:00
|
|
|
this.animatingOut = false;
|
2017-12-05 17:04:24 -05:00
|
|
|
|
2018-07-21 01:08:33 -04:00
|
|
|
this.connect('notify::scale-x', () => this.queue_relayout());
|
|
|
|
this.connect('notify::scale-y', () => this.queue_relayout());
|
|
|
|
|
2017-12-05 17:04:24 -05:00
|
|
|
this.connect('destroy', () => {
|
2018-11-16 13:31:56 -05:00
|
|
|
if (this.child != null)
|
|
|
|
this.child.destroy();
|
2017-12-05 17:04:24 -05:00
|
|
|
this.label.destroy();
|
|
|
|
});
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
vfunc_get_preferred_height(forWidth) {
|
2012-12-22 08:57:06 -05:00
|
|
|
let themeNode = this.get_theme_node();
|
|
|
|
forWidth = themeNode.adjust_for_width(forWidth);
|
2017-10-30 21:23:39 -04:00
|
|
|
let [minHeight, natHeight] = super.vfunc_get_preferred_height(forWidth);
|
2018-08-23 11:17:21 -04:00
|
|
|
return themeNode.adjust_preferred_height(minHeight * this.scale_y,
|
|
|
|
natHeight * this.scale_y);
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
vfunc_get_preferred_width(forHeight) {
|
2012-12-22 08:57:06 -05:00
|
|
|
let themeNode = this.get_theme_node();
|
|
|
|
forHeight = themeNode.adjust_for_height(forHeight);
|
2017-10-30 21:23:39 -04:00
|
|
|
let [minWidth, natWidth] = super.vfunc_get_preferred_width(forHeight);
|
2018-08-23 11:17:21 -04:00
|
|
|
return themeNode.adjust_preferred_width(minWidth * this.scale_x,
|
|
|
|
natWidth * this.scale_x);
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
showLabel() {
|
2012-10-03 01:22:46 -04:00
|
|
|
if (!this._labelText)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this.label.set_text(this._labelText);
|
2012-03-15 15:18:53 -04:00
|
|
|
this.label.opacity = 0;
|
|
|
|
this.label.show();
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2012-12-22 08:57:06 -05:00
|
|
|
let [stageX, stageY] = this.get_transformed_position();
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
const itemWidth = this.allocation.get_width();
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
const labelWidth = this.label.get_width();
|
|
|
|
const xOffset = Math.floor((itemWidth - labelWidth) / 2);
|
|
|
|
const x = Math.clamp(stageX + xOffset, 0, global.stage.width - labelWidth);
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2012-03-15 15:18:53 -04:00
|
|
|
let node = this.label.get_theme_node();
|
2020-12-11 07:47:28 -05:00
|
|
|
const yOffset = node.get_length('-y-offset');
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
const y = stageY - this.label.height - yOffset;
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2012-03-15 15:18:53 -04:00
|
|
|
this.label.set_position(x, y);
|
2018-07-20 15:46:19 -04:00
|
|
|
this.label.ease({
|
|
|
|
opacity: 255,
|
|
|
|
duration: DASH_ITEM_LABEL_SHOW_TIME,
|
2019-08-20 17:43:54 -04:00
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2018-07-20 15:46:19 -04:00
|
|
|
});
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
setLabelText(text) {
|
2012-10-03 01:22:46 -04:00
|
|
|
this._labelText = text;
|
2012-10-24 13:29:07 -04:00
|
|
|
this.child.accessible_name = text;
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
hideLabel() {
|
2018-07-20 15:46:19 -04:00
|
|
|
this.label.ease({
|
|
|
|
opacity: 0,
|
|
|
|
duration: DASH_ITEM_LABEL_HIDE_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2019-08-20 17:43:54 -04:00
|
|
|
onComplete: () => this.label.hide(),
|
2018-07-20 15:46:19 -04:00
|
|
|
});
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
setChild(actor) {
|
2011-01-20 10:23:43 -05:00
|
|
|
if (this.child == actor)
|
|
|
|
return;
|
|
|
|
|
2012-12-22 08:57:06 -05:00
|
|
|
this.destroy_all_children();
|
2011-01-20 10:23:43 -05:00
|
|
|
|
|
|
|
this.child = actor;
|
2012-12-22 08:57:06 -05:00
|
|
|
this.add_actor(this.child);
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-01-21 06:06:13 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
show(animate) {
|
2011-01-21 06:06:13 -05:00
|
|
|
if (this.child == null)
|
|
|
|
return;
|
|
|
|
|
2012-12-22 09:26:12 -05:00
|
|
|
let time = animate ? DASH_ANIMATION_TIME : 0;
|
2018-07-20 15:46:19 -04:00
|
|
|
this.ease({
|
|
|
|
scale_x: 1,
|
|
|
|
scale_y: 1,
|
|
|
|
opacity: 255,
|
|
|
|
duration: time,
|
2019-08-20 17:43:54 -04:00
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2018-07-20 15:46:19 -04:00
|
|
|
});
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-01-21 06:06:13 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
animateOutAndDestroy() {
|
2017-12-04 20:45:35 -05:00
|
|
|
this.label.hide();
|
2012-04-08 15:35:41 -04:00
|
|
|
|
2011-01-21 06:06:13 -05:00
|
|
|
if (this.child == null) {
|
2012-12-22 08:57:06 -05:00
|
|
|
this.destroy();
|
2011-01-21 06:06:13 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-09-19 17:47:16 -04:00
|
|
|
this.animatingOut = true;
|
2018-07-20 15:46:19 -04:00
|
|
|
this.ease({
|
|
|
|
scale_x: 0,
|
|
|
|
scale_y: 0,
|
|
|
|
opacity: 0,
|
|
|
|
duration: DASH_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2019-08-20 17:43:54 -04:00
|
|
|
onComplete: () => this.destroy(),
|
2018-07-20 15:46:19 -04:00
|
|
|
});
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2011-11-20 11:07:14 -05:00
|
|
|
});
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 21:23:39 -04:00
|
|
|
var ShowAppsIcon = GObject.registerClass(
|
|
|
|
class ShowAppsIcon extends DashItemContainer {
|
2017-10-30 20:03:21 -04:00
|
|
|
_init() {
|
2017-10-30 21:23:39 -04:00
|
|
|
super._init();
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2012-07-22 08:45:53 -04:00
|
|
|
this.toggleButton = new St.Button({ style_class: 'show-apps',
|
|
|
|
track_hover: true,
|
|
|
|
can_focus: true,
|
|
|
|
toggle_mode: true });
|
2010-11-07 20:51:02 -05:00
|
|
|
this._iconActor = null;
|
2012-08-14 10:34:14 -04:00
|
|
|
this.icon = new IconGrid.BaseIcon(_("Show Applications"),
|
2019-01-29 14:36:54 -05:00
|
|
|
{ setSizeManually: true,
|
|
|
|
showLabel: false,
|
|
|
|
createIcon: this._createIcon.bind(this) });
|
2018-06-30 14:00:08 -04:00
|
|
|
this.toggleButton.add_actor(this.icon);
|
2012-07-22 08:45:53 -04:00
|
|
|
this.toggleButton._delegate = this;
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2012-07-22 08:45:53 -04:00
|
|
|
this.setChild(this.toggleButton);
|
2012-10-03 01:17:07 -04:00
|
|
|
this.setDragApp(null);
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_createIcon(size) {
|
2016-02-09 09:05:41 -05:00
|
|
|
this._iconActor = new St.Icon({ icon_name: 'view-app-grid-symbolic',
|
2012-05-30 09:58:37 -04:00
|
|
|
icon_size: size,
|
2012-08-14 10:34:14 -04:00
|
|
|
style_class: 'show-apps-icon',
|
2012-05-30 09:58:37 -04:00
|
|
|
track_hover: true });
|
2010-11-07 20:51:02 -05:00
|
|
|
return this._iconActor;
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_canRemoveApp(app) {
|
2012-10-03 01:17:07 -04:00
|
|
|
if (app == null)
|
|
|
|
return false;
|
|
|
|
|
2015-03-20 15:56:52 -04:00
|
|
|
if (!global.settings.is_writable('favorite-apps'))
|
2014-12-11 10:22:19 -05:00
|
|
|
return false;
|
|
|
|
|
2012-10-03 01:17:07 -04:00
|
|
|
let id = app.get_id();
|
|
|
|
let isFavorite = AppFavorites.getAppFavorites().isFavorite(id);
|
|
|
|
return isFavorite;
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2012-10-03 01:17:07 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
setDragApp(app) {
|
2012-10-03 01:17:07 -04:00
|
|
|
let canRemove = this._canRemoveApp(app);
|
|
|
|
|
|
|
|
this.toggleButton.set_hover(canRemove);
|
2010-11-07 20:51:02 -05:00
|
|
|
if (this._iconActor)
|
2012-10-03 01:17:07 -04:00
|
|
|
this._iconActor.set_hover(canRemove);
|
2012-08-22 05:46:28 -04:00
|
|
|
|
2012-10-03 01:17:07 -04:00
|
|
|
if (canRemove)
|
2012-08-22 05:46:28 -04:00
|
|
|
this.setLabelText(_("Remove from Favorites"));
|
|
|
|
else
|
|
|
|
this.setLabelText(_("Show Applications"));
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2019-01-31 09:08:10 -05:00
|
|
|
handleDragOver(source, _actor, _x, _y, _time) {
|
2013-05-31 14:00:17 -04:00
|
|
|
if (!this._canRemoveApp(getAppFromSource(source)))
|
2012-10-02 19:53:07 -04:00
|
|
|
return DND.DragMotionResult.NO_DROP;
|
|
|
|
|
2010-11-07 20:51:02 -05:00
|
|
|
return DND.DragMotionResult.MOVE_DROP;
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2019-01-31 09:08:10 -05:00
|
|
|
acceptDrop(source, _actor, _x, _y, _time) {
|
2012-10-02 20:13:47 -04:00
|
|
|
let app = getAppFromSource(source);
|
2013-05-31 14:00:17 -04:00
|
|
|
if (!this._canRemoveApp(app))
|
2012-10-02 20:13:47 -04:00
|
|
|
return false;
|
2010-11-07 20:51:02 -05:00
|
|
|
|
|
|
|
let id = app.get_id();
|
|
|
|
|
2017-10-30 20:38:18 -04:00
|
|
|
Meta.later_add(Meta.LaterType.BEFORE_REDRAW, () => {
|
|
|
|
AppFavorites.getAppFavorites().removeFavorite(id);
|
|
|
|
return false;
|
|
|
|
});
|
2010-11-07 20:51:02 -05:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2011-11-20 11:07:14 -05:00
|
|
|
});
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 21:23:39 -04:00
|
|
|
var DragPlaceholderItem = GObject.registerClass(
|
|
|
|
class DragPlaceholderItem extends DashItemContainer {
|
2017-10-30 20:03:21 -04:00
|
|
|
_init() {
|
2017-10-30 21:23:39 -04:00
|
|
|
super._init();
|
2011-05-04 22:47:07 -04:00
|
|
|
this.setChild(new St.Bin({ style_class: 'placeholder' }));
|
2011-01-20 10:23:43 -05:00
|
|
|
}
|
2011-11-20 11:07:14 -05:00
|
|
|
});
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2017-10-30 21:23:39 -04:00
|
|
|
var EmptyDropTargetItem = GObject.registerClass(
|
|
|
|
class EmptyDropTargetItem extends DashItemContainer {
|
2017-10-30 20:03:21 -04:00
|
|
|
_init() {
|
2017-10-30 21:23:39 -04:00
|
|
|
super._init();
|
2013-05-31 11:59:27 -04:00
|
|
|
this.setChild(new St.Bin({ style_class: 'empty-dash-drop-target' }));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-01-19 11:34:19 -05:00
|
|
|
const DashIconsLayout = GObject.registerClass(
|
|
|
|
class DashIconsLayout extends Clutter.BoxLayout {
|
2017-10-30 20:03:21 -04:00
|
|
|
_init() {
|
2019-10-17 17:40:24 -04:00
|
|
|
super._init({
|
2021-01-19 11:34:19 -05:00
|
|
|
orientation: Clutter.Orientation.HORIZONTAL,
|
2019-10-17 17:40:24 -04:00
|
|
|
});
|
2017-10-30 21:23:39 -04:00
|
|
|
}
|
2012-09-10 14:42:08 -04:00
|
|
|
|
2021-01-19 11:34:19 -05:00
|
|
|
vfunc_get_preferred_width(container, forHeight) {
|
|
|
|
const [, natWidth] = super.vfunc_get_preferred_width(container, forHeight);
|
|
|
|
return [0, natWidth];
|
2012-09-10 14:42:08 -04:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-01-28 20:27:05 -05:00
|
|
|
const baseIconSizes = [16, 22, 24, 32, 48, 64];
|
2014-02-16 00:35:13 -05:00
|
|
|
|
2019-07-16 05:24:13 -04:00
|
|
|
var Dash = GObject.registerClass({
|
2019-08-20 17:43:54 -04:00
|
|
|
Signals: { 'icon-size-changed': {} },
|
2021-01-19 11:34:19 -05:00
|
|
|
}, class Dash extends St.BoxLayout {
|
2019-07-16 05:24:13 -04:00
|
|
|
_init() {
|
2020-12-11 07:47:28 -05:00
|
|
|
this._maxWidth = -1;
|
2011-02-11 19:17:10 -05:00
|
|
|
this.iconSize = 64;
|
2011-02-02 10:44:07 -05:00
|
|
|
this._shownInitially = false;
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2021-01-28 17:24:44 -05:00
|
|
|
this._separator = null;
|
2010-11-07 20:51:02 -05:00
|
|
|
this._dragPlaceholder = null;
|
|
|
|
this._dragPlaceholderPos = -1;
|
2011-01-21 06:06:13 -05:00
|
|
|
this._animatingPlaceholdersCount = 0;
|
2012-11-06 18:33:32 -05:00
|
|
|
this._showLabelTimeoutId = 0;
|
2012-01-20 10:14:16 -05:00
|
|
|
this._resetHoverTimeoutId = 0;
|
2012-11-06 18:33:32 -05:00
|
|
|
this._labelShowing = false;
|
2010-07-21 19:29:02 -04:00
|
|
|
|
2021-01-19 11:34:19 -05:00
|
|
|
super._init({
|
|
|
|
name: 'dash',
|
|
|
|
offscreen_redirect: Clutter.OffscreenRedirect.ALWAYS,
|
|
|
|
x_align: Clutter.ActorAlign.CENTER,
|
|
|
|
});
|
|
|
|
|
|
|
|
this._box = new St.Widget({
|
|
|
|
clip_to_allocation: true,
|
|
|
|
layout_manager: new DashIconsLayout(),
|
|
|
|
x_expand: true,
|
|
|
|
x_align: Clutter.ActorAlign.CENTER,
|
|
|
|
});
|
2010-07-21 19:29:02 -04:00
|
|
|
this._box._delegate = this;
|
2021-01-19 11:34:19 -05:00
|
|
|
this.add_child(this._box);
|
2012-08-14 10:27:28 -04:00
|
|
|
|
2012-08-14 10:34:14 -04:00
|
|
|
this._showAppsIcon = new ShowAppsIcon();
|
2018-07-21 01:08:33 -04:00
|
|
|
this._showAppsIcon.show(false);
|
2012-08-14 10:34:14 -04:00
|
|
|
this._showAppsIcon.icon.setIconSize(this.iconSize);
|
2012-11-06 18:33:32 -05:00
|
|
|
this._hookUpLabel(this._showAppsIcon);
|
2021-01-19 11:34:19 -05:00
|
|
|
this.add_child(this._showAppsIcon);
|
2012-08-14 10:27:28 -04:00
|
|
|
|
2012-07-22 08:45:53 -04:00
|
|
|
this.showAppsButton = this._showAppsIcon.toggleButton;
|
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
this.connect('notify::width', () => {
|
|
|
|
if (this._maxWidth !== this.width)
|
2017-10-30 20:38:18 -04:00
|
|
|
this._queueRedisplay();
|
2020-12-11 07:47:28 -05:00
|
|
|
this._maxWidth = this.width;
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2010-07-21 19:29:02 -04:00
|
|
|
|
2017-12-01 19:27:35 -05:00
|
|
|
this._workId = Main.initializeDeferredWork(this._box, this._redisplay.bind(this));
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
this._appSystem = Shell.AppSystem.get_default();
|
|
|
|
|
2017-10-30 20:38:18 -04:00
|
|
|
this._appSystem.connect('installed-changed', () => {
|
2013-08-27 08:00:26 -04:00
|
|
|
AppFavorites.getAppFavorites().reload();
|
2013-08-31 13:52:21 -04:00
|
|
|
this._queueRedisplay();
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2017-12-01 19:27:35 -05:00
|
|
|
AppFavorites.getAppFavorites().connect('changed', this._queueRedisplay.bind(this));
|
|
|
|
this._appSystem.connect('app-state-changed', this._queueRedisplay.bind(this));
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2011-02-11 18:01:59 -05:00
|
|
|
Main.overview.connect('item-drag-begin',
|
2017-12-01 19:27:35 -05:00
|
|
|
this._onDragBegin.bind(this));
|
2011-02-11 18:01:59 -05:00
|
|
|
Main.overview.connect('item-drag-end',
|
2017-12-01 19:27:35 -05:00
|
|
|
this._onDragEnd.bind(this));
|
2011-03-09 10:56:08 -05:00
|
|
|
Main.overview.connect('item-drag-cancelled',
|
2017-12-01 19:27:35 -05:00
|
|
|
this._onDragCancelled.bind(this));
|
2013-02-15 18:23:15 -05:00
|
|
|
|
|
|
|
// Translators: this is the name of the dock/favorites area on
|
|
|
|
// the left of the overview
|
2019-07-16 05:24:13 -04:00
|
|
|
Main.ctrlAltTabManager.addGroup(this, _("Dash"), 'user-bookmarks-symbolic');
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onDragBegin() {
|
2011-03-09 10:56:08 -05:00
|
|
|
this._dragCancelled = false;
|
2010-11-07 20:51:02 -05:00
|
|
|
this._dragMonitor = {
|
2019-08-20 17:43:54 -04:00
|
|
|
dragMotion: this._onDragMotion.bind(this),
|
2010-11-07 20:51:02 -05:00
|
|
|
};
|
|
|
|
DND.addDragMonitor(this._dragMonitor);
|
2013-05-31 11:59:27 -04:00
|
|
|
|
|
|
|
if (this._box.get_n_children() == 0) {
|
|
|
|
this._emptyDropTarget = new EmptyDropTargetItem();
|
|
|
|
this._box.insert_child_at_index(this._emptyDropTarget, 0);
|
|
|
|
this._emptyDropTarget.show(true);
|
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onDragCancelled() {
|
2011-03-09 10:56:08 -05:00
|
|
|
this._dragCancelled = true;
|
|
|
|
this._endDrag();
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-03-09 10:56:08 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onDragEnd() {
|
2011-03-09 10:56:08 -05:00
|
|
|
if (this._dragCancelled)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this._endDrag();
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-03-09 10:56:08 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_endDrag() {
|
2010-11-07 20:51:02 -05:00
|
|
|
this._clearDragPlaceholder();
|
2013-05-31 11:59:27 -04:00
|
|
|
this._clearEmptyDropTarget();
|
2012-10-08 14:33:21 -04:00
|
|
|
this._showAppsIcon.setDragApp(null);
|
2011-06-27 12:59:56 -04:00
|
|
|
DND.removeDragMonitor(this._dragMonitor);
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onDragMotion(dragEvent) {
|
2012-10-02 20:13:47 -04:00
|
|
|
let app = getAppFromSource(dragEvent.source);
|
|
|
|
if (app == null)
|
2010-11-07 20:51:02 -05:00
|
|
|
return DND.DragMotionResult.CONTINUE;
|
|
|
|
|
2012-08-14 10:34:14 -04:00
|
|
|
let showAppsHovered =
|
2012-12-22 08:57:06 -05:00
|
|
|
this._showAppsIcon.contains(dragEvent.targetActor);
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2012-08-14 10:34:14 -04:00
|
|
|
if (!this._box.contains(dragEvent.targetActor) || showAppsHovered)
|
2010-11-07 20:51:02 -05:00
|
|
|
this._clearDragPlaceholder();
|
|
|
|
|
2012-10-03 01:17:07 -04:00
|
|
|
if (showAppsHovered)
|
|
|
|
this._showAppsIcon.setDragApp(app);
|
|
|
|
else
|
|
|
|
this._showAppsIcon.setDragApp(null);
|
2010-11-07 20:51:02 -05:00
|
|
|
|
|
|
|
return DND.DragMotionResult.CONTINUE;
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_appIdListToHash(apps) {
|
2012-11-06 18:33:32 -05:00
|
|
|
let ids = {};
|
|
|
|
for (let i = 0; i < apps.length; i++)
|
|
|
|
ids[apps[i].get_id()] = apps[i];
|
|
|
|
return ids;
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2012-11-06 18:33:32 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_queueRedisplay() {
|
2010-07-21 19:29:02 -04:00
|
|
|
Main.queueDeferredWork(this._workId);
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-07-21 19:29:02 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_hookUpLabel(item, appIcon) {
|
2017-10-30 20:38:18 -04:00
|
|
|
item.child.connect('notify::hover', () => {
|
2013-08-18 10:20:22 -04:00
|
|
|
this._syncLabel(item, appIcon);
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2012-11-06 18:33:32 -05:00
|
|
|
|
2020-08-27 04:52:14 -04:00
|
|
|
item.child.connect('clicked', () => {
|
|
|
|
this._labelShowing = false;
|
|
|
|
item.hideLabel();
|
|
|
|
});
|
|
|
|
|
2017-10-30 20:38:18 -04:00
|
|
|
let id = Main.overview.connect('hiding', () => {
|
2012-11-06 18:33:32 -05:00
|
|
|
this._labelShowing = false;
|
|
|
|
item.hideLabel();
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
|
|
|
item.child.connect('destroy', () => {
|
2015-02-26 20:09:02 -05:00
|
|
|
Main.overview.disconnect(id);
|
|
|
|
});
|
2013-08-18 10:20:22 -04:00
|
|
|
|
|
|
|
if (appIcon) {
|
2017-10-30 20:38:18 -04:00
|
|
|
appIcon.connect('sync-tooltip', () => {
|
2013-08-18 10:20:22 -04:00
|
|
|
this._syncLabel(item, appIcon);
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2013-08-18 10:20:22 -04:00
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2012-08-22 05:46:28 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_createAppItem(app) {
|
2019-07-12 18:32:54 -04:00
|
|
|
let appIcon = new DashIcon(app);
|
2015-01-27 16:10:45 -05:00
|
|
|
|
2012-11-03 16:03:49 -04:00
|
|
|
appIcon.connect('menu-state-changed',
|
2019-08-19 20:20:08 -04:00
|
|
|
(o, opened) => {
|
2012-11-03 16:03:33 -04:00
|
|
|
this._itemMenuStateChanged(item, opened);
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2011-02-09 19:41:51 -05:00
|
|
|
|
2011-01-20 10:23:43 -05:00
|
|
|
let item = new DashItemContainer();
|
2019-07-16 05:24:13 -04:00
|
|
|
item.setChild(appIcon);
|
2011-01-20 10:23:43 -05:00
|
|
|
|
2013-01-30 17:51:43 -05:00
|
|
|
// Override default AppIcon label_actor, now the
|
2012-10-24 13:29:07 -04:00
|
|
|
// accessible_name is set at DashItemContainer.setLabelText
|
2019-07-16 05:24:13 -04:00
|
|
|
appIcon.label_actor = null;
|
2011-12-16 17:46:07 -05:00
|
|
|
item.setLabelText(app.get_name());
|
2012-10-24 13:29:07 -04:00
|
|
|
|
2012-11-03 16:03:49 -04:00
|
|
|
appIcon.icon.setIconSize(this.iconSize);
|
2013-08-18 10:20:22 -04:00
|
|
|
this._hookUpLabel(item, appIcon);
|
2012-04-16 11:04:29 -04:00
|
|
|
|
2011-01-20 10:23:43 -05:00
|
|
|
return item;
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_itemMenuStateChanged(item, opened) {
|
2012-11-03 16:03:33 -04:00
|
|
|
// When the menu closes, it calls sync_hover, which means
|
|
|
|
// that the notify::hover handler does everything we need to.
|
|
|
|
if (opened) {
|
2012-11-06 18:33:32 -05:00
|
|
|
if (this._showLabelTimeoutId > 0) {
|
2019-08-19 14:50:33 -04:00
|
|
|
GLib.source_remove(this._showLabelTimeoutId);
|
2012-11-06 18:33:32 -05:00
|
|
|
this._showLabelTimeoutId = 0;
|
2012-11-03 16:03:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
item.hideLabel();
|
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2012-11-03 16:03:33 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_syncLabel(item, appIcon) {
|
2013-08-18 10:20:22 -04:00
|
|
|
let shouldShow = appIcon ? appIcon.shouldShowTooltip() : item.child.get_hover();
|
|
|
|
|
|
|
|
if (shouldShow) {
|
2012-11-06 18:33:32 -05:00
|
|
|
if (this._showLabelTimeoutId == 0) {
|
|
|
|
let timeout = this._labelShowing ? 0 : DASH_ITEM_HOVER_TIMEOUT;
|
2019-08-19 14:50:33 -04:00
|
|
|
this._showLabelTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, timeout,
|
2017-10-30 20:38:18 -04:00
|
|
|
() => {
|
2012-11-06 18:33:32 -05:00
|
|
|
this._labelShowing = true;
|
2011-12-16 17:46:07 -05:00
|
|
|
item.showLabel();
|
2013-11-09 11:11:01 -05:00
|
|
|
this._showLabelTimeoutId = 0;
|
2013-11-28 19:45:39 -05:00
|
|
|
return GLib.SOURCE_REMOVE;
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2014-04-10 13:26:52 -04:00
|
|
|
GLib.Source.set_name_by_id(this._showLabelTimeoutId, '[gnome-shell] item.showLabel');
|
2012-01-20 10:14:16 -05:00
|
|
|
if (this._resetHoverTimeoutId > 0) {
|
2019-08-19 14:50:33 -04:00
|
|
|
GLib.source_remove(this._resetHoverTimeoutId);
|
2012-04-16 11:04:29 -04:00
|
|
|
this._resetHoverTimeoutId = 0;
|
2012-01-20 10:14:16 -05:00
|
|
|
}
|
2011-12-16 17:46:07 -05:00
|
|
|
}
|
|
|
|
} else {
|
2012-11-06 18:33:32 -05:00
|
|
|
if (this._showLabelTimeoutId > 0)
|
2019-08-19 14:50:33 -04:00
|
|
|
GLib.source_remove(this._showLabelTimeoutId);
|
2012-11-06 18:33:32 -05:00
|
|
|
this._showLabelTimeoutId = 0;
|
2011-12-16 17:46:07 -05:00
|
|
|
item.hideLabel();
|
2012-11-06 18:33:32 -05:00
|
|
|
if (this._labelShowing) {
|
2019-08-19 14:50:33 -04:00
|
|
|
this._resetHoverTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, DASH_ITEM_HOVER_TIMEOUT,
|
2017-10-30 20:38:18 -04:00
|
|
|
() => {
|
2012-11-06 18:33:32 -05:00
|
|
|
this._labelShowing = false;
|
2013-11-09 11:11:01 -05:00
|
|
|
this._resetHoverTimeoutId = 0;
|
2013-11-28 19:45:39 -05:00
|
|
|
return GLib.SOURCE_REMOVE;
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2014-04-10 13:26:52 -04:00
|
|
|
GLib.Source.set_name_by_id(this._resetHoverTimeoutId, '[gnome-shell] this._labelShowing');
|
2012-01-20 10:14:16 -05:00
|
|
|
}
|
2011-12-16 17:46:07 -05:00
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-12-16 17:46:07 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_adjustIconSize() {
|
2011-09-20 11:59:25 -04:00
|
|
|
// For the icon size, we only consider children which are "proper"
|
|
|
|
// icons (i.e. ignoring drag placeholders) and which are not
|
|
|
|
// animating out (which means they will be destroyed at the end of
|
|
|
|
// the animation)
|
2017-10-30 20:38:18 -04:00
|
|
|
let iconChildren = this._box.get_children().filter(actor => {
|
2012-12-22 08:57:06 -05:00
|
|
|
return actor.child &&
|
|
|
|
actor.child._delegate &&
|
|
|
|
actor.child._delegate.icon &&
|
|
|
|
!actor.animatingOut;
|
2011-09-20 11:59:25 -04:00
|
|
|
});
|
|
|
|
|
2012-12-22 08:57:06 -05:00
|
|
|
iconChildren.push(this._showAppsIcon);
|
2010-11-27 16:31:14 -05:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
if (this._maxWidth === -1)
|
2010-11-27 16:31:14 -05:00
|
|
|
return;
|
|
|
|
|
2021-01-19 11:34:19 -05:00
|
|
|
const themeNode = this.get_theme_node();
|
2020-12-11 07:47:28 -05:00
|
|
|
const maxAllocation = new Clutter.ActorBox({
|
|
|
|
x1: 0,
|
|
|
|
y1: 0,
|
|
|
|
x2: this._maxWidth,
|
|
|
|
y2: 42, /* whatever */
|
|
|
|
});
|
2011-09-20 11:59:25 -04:00
|
|
|
let maxContent = themeNode.get_content_box(maxAllocation);
|
2020-12-11 07:47:28 -05:00
|
|
|
let availWidth = maxContent.x2 - maxContent.x1;
|
2011-09-20 11:59:25 -04:00
|
|
|
let spacing = themeNode.get_length('spacing');
|
2011-09-19 17:47:16 -04:00
|
|
|
|
2012-12-22 09:26:12 -05:00
|
|
|
let firstButton = iconChildren[0].child;
|
|
|
|
let firstIcon = firstButton._delegate.icon;
|
2011-09-20 11:59:25 -04:00
|
|
|
|
dash: Fix messed up icon height
When determining the biggest icon size that fits the available height,
we first subtract the additional space requirements of icons (spacing,
padding, running indicator etc.) and then divide the result by the
number of icons to get the maximum size available to each icon texture.
In the above, the additional space requirement of each icon is taken
from the first icon (as all icons are assumed to be the same), and
calculated as the difference between the icon button's preferred height
and the currently used icon size.
To make sure that the icon is actually using the dash's current icon
size (even while animating to a new icon size), we enforce its height
during the size request and restore its original height afterwards.
However after some recent changes, that step is causing troubles:
For some reason, the original height may be 0, and when we restore it,
we end up forcing a fixed non-height that bypasses the regular size
request machinery.
While it is unclear where exactly the zero height comes from (maybe
waiting for a valid resource scale?), it is clear that it's best
to avoid forcing a fixed height. So instead of making the icon
texture comply with the assumed icon size, adjust the calculations
to use its current height request.
https://gitlab.gnome.org/GNOME/gnome-shell/issues/1053
2019-04-08 21:03:54 -04:00
|
|
|
// Enforce valid spacings during the size request
|
2015-10-13 11:38:26 -04:00
|
|
|
firstIcon.icon.ensure_style();
|
2020-12-11 07:47:28 -05:00
|
|
|
let [, iconWidth] = firstIcon.icon.get_preferred_width(-1);
|
|
|
|
let [, buttonWidth] = firstButton.get_preferred_width(-1);
|
2010-11-27 16:31:14 -05:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
// Subtract icon padding and box spacing from the available width
|
|
|
|
availWidth -= iconChildren.length * (buttonWidth - iconWidth) +
|
2011-09-20 11:59:25 -04:00
|
|
|
(iconChildren.length - 1) * spacing;
|
2010-11-27 16:31:14 -05:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
let availSize = availWidth / iconChildren.length;
|
2011-09-19 17:47:16 -04:00
|
|
|
|
dash: Fix messed up icon height
When determining the biggest icon size that fits the available height,
we first subtract the additional space requirements of icons (spacing,
padding, running indicator etc.) and then divide the result by the
number of icons to get the maximum size available to each icon texture.
In the above, the additional space requirement of each icon is taken
from the first icon (as all icons are assumed to be the same), and
calculated as the difference between the icon button's preferred height
and the currently used icon size.
To make sure that the icon is actually using the dash's current icon
size (even while animating to a new icon size), we enforce its height
during the size request and restore its original height afterwards.
However after some recent changes, that step is causing troubles:
For some reason, the original height may be 0, and when we restore it,
we end up forcing a fixed non-height that bypasses the regular size
request machinery.
While it is unclear where exactly the zero height comes from (maybe
waiting for a valid resource scale?), it is clear that it's best
to avoid forcing a fixed height. So instead of making the icon
texture comply with the assumed icon size, adjust the calculations
to use its current height request.
https://gitlab.gnome.org/GNOME/gnome-shell/issues/1053
2019-04-08 21:03:54 -04:00
|
|
|
let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor;
|
2017-10-30 20:38:18 -04:00
|
|
|
let iconSizes = baseIconSizes.map(s => s * scaleFactor);
|
2015-10-03 17:44:48 -04:00
|
|
|
|
2014-02-16 00:35:13 -05:00
|
|
|
let newIconSize = baseIconSizes[0];
|
2010-11-27 16:31:14 -05:00
|
|
|
for (let i = 0; i < iconSizes.length; i++) {
|
2020-12-10 17:21:30 -05:00
|
|
|
if (iconSizes[i] <= availSize)
|
2014-02-16 00:35:13 -05:00
|
|
|
newIconSize = baseIconSizes[i];
|
2010-11-27 16:31:14 -05:00
|
|
|
}
|
|
|
|
|
2011-02-11 19:17:10 -05:00
|
|
|
if (newIconSize == this.iconSize)
|
2010-11-27 16:31:14 -05:00
|
|
|
return;
|
|
|
|
|
2011-02-11 19:17:10 -05:00
|
|
|
let oldIconSize = this.iconSize;
|
|
|
|
this.iconSize = newIconSize;
|
2011-08-28 09:35:13 -04:00
|
|
|
this.emit('icon-size-changed');
|
2010-11-27 16:31:14 -05:00
|
|
|
|
2011-01-21 06:06:13 -05:00
|
|
|
let scale = oldIconSize / newIconSize;
|
2010-11-27 16:31:14 -05:00
|
|
|
for (let i = 0; i < iconChildren.length; i++) {
|
2012-12-22 08:57:06 -05:00
|
|
|
let icon = iconChildren[i].child._delegate.icon;
|
2011-01-21 06:06:13 -05:00
|
|
|
|
|
|
|
// Set the new size immediately, to keep the icons' sizes
|
2011-02-11 19:17:10 -05:00
|
|
|
// in sync with this.iconSize
|
|
|
|
icon.setIconSize(this.iconSize);
|
2011-02-09 17:55:50 -05:00
|
|
|
|
|
|
|
// Don't animate the icon size change when the overview
|
2012-10-30 15:15:29 -04:00
|
|
|
// is transitioning, not visible or when initially filling
|
|
|
|
// the dash
|
|
|
|
if (!Main.overview.visible || Main.overview.animationInProgress ||
|
|
|
|
!this._shownInitially)
|
2011-02-09 17:55:50 -05:00
|
|
|
continue;
|
|
|
|
|
2011-01-21 06:06:13 -05:00
|
|
|
let [targetWidth, targetHeight] = icon.icon.get_size();
|
|
|
|
|
|
|
|
// Scale the icon's texture to the previous size and
|
|
|
|
// tween to the new size
|
|
|
|
icon.icon.set_size(icon.icon.width * scale,
|
|
|
|
icon.icon.height * scale);
|
|
|
|
|
2018-07-20 15:46:19 -04:00
|
|
|
icon.icon.ease({
|
|
|
|
width: targetWidth,
|
|
|
|
height: targetHeight,
|
2019-10-09 20:14:28 -04:00
|
|
|
duration: DASH_ANIMATION_TIME,
|
2019-08-20 17:43:54 -04:00
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
2018-07-20 15:46:19 -04:00
|
|
|
});
|
2010-11-27 16:31:14 -05:00
|
|
|
}
|
2021-01-28 17:24:44 -05:00
|
|
|
|
|
|
|
if (this._separator) {
|
|
|
|
this._separator.ease({
|
|
|
|
height: this.iconSize,
|
|
|
|
duration: DASH_ANIMATION_TIME,
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
});
|
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-27 16:31:14 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_redisplay() {
|
2010-07-21 19:29:02 -04:00
|
|
|
let favorites = AppFavorites.getAppFavorites().getFavoriteMap();
|
|
|
|
|
apps: Ensure running apps override new .desktop file data
This patch fixes the "apps vanish from alt-TAB bug".
If a "package system" rips away and possibly replaces .desktop files
at some random time, we have historically used inotify to detect this
and reread state (in a racy way, but...). In GNOME 2, this was
generally not too problematic because the menu widget was totally
separate from the list of windows - and the data they operate on was
disjoint as well.
In GNOME 3 we unify these, and this creates architectural problems
because the windows are tied to the app.
What this patch tries to do is, when rereading the application state,
if we have a running application, we keep that app around instead of
making a new instance. This ensures we preserve any state such as the
set of open windows.
This requires moving the running state into ShellAppSystem. Adjust
callers as necessary, and while we're at it drop the unused "contexts"
stuff.
This is just a somewhat quick band-aid; a REAL fix would require us
having low-level control over application installation. As long as
we're on top of random broken tar+wget wrappers, it will be gross.
A slight future improvement to this patch would add an explicit
"merge" between the old and new data. I think probably we always keep
around the ShellApp corresponding to a given ID, but replace its
GMenuTreeEntry.
https://bugzilla.gnome.org/show_bug.cgi?id=657990
2011-09-03 10:32:06 -04:00
|
|
|
let running = this._appSystem.get_running();
|
2010-07-21 19:29:02 -04:00
|
|
|
|
2017-10-30 20:38:18 -04:00
|
|
|
let children = this._box.get_children().filter(actor => {
|
2019-01-29 14:36:54 -05:00
|
|
|
return actor.child &&
|
|
|
|
actor.child._delegate &&
|
|
|
|
actor.child._delegate.app;
|
|
|
|
});
|
2011-01-13 07:44:40 -05:00
|
|
|
// Apps currently in the dash
|
2017-10-30 20:38:18 -04:00
|
|
|
let oldApps = children.map(actor => actor.child._delegate.app);
|
2011-01-13 07:44:40 -05:00
|
|
|
// Apps supposed to be in the dash
|
|
|
|
let newApps = [];
|
|
|
|
|
|
|
|
for (let id in favorites)
|
|
|
|
newApps.push(favorites[id]);
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
for (let i = 0; i < running.length; i++) {
|
|
|
|
let app = running[i];
|
|
|
|
if (app.get_id() in favorites)
|
|
|
|
continue;
|
2011-01-13 07:44:40 -05:00
|
|
|
newApps.push(app);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Figure out the actual changes to the list of items; we iterate
|
|
|
|
// over both the list of items currently in the dash and the list
|
|
|
|
// of items expected there, and collect additions and removals.
|
|
|
|
// Moves are both an addition and a removal, where the order of
|
|
|
|
// the operations depends on whether we encounter the position
|
|
|
|
// where the item has been added first or the one from where it
|
|
|
|
// was removed.
|
|
|
|
// There is an assumption that only one item is moved at a given
|
|
|
|
// time; when moving several items at once, everything will still
|
|
|
|
// end up at the right position, but there might be additional
|
|
|
|
// additions/removals (e.g. it might remove all the launchers
|
|
|
|
// and add them back in the new order even if a smaller set of
|
|
|
|
// additions and removals is possible).
|
|
|
|
// If above assumptions turns out to be a problem, we might need
|
|
|
|
// to use a more sophisticated algorithm, e.g. Longest Common
|
|
|
|
// Subsequence as used by diff.
|
|
|
|
let addedItems = [];
|
|
|
|
let removedActors = [];
|
|
|
|
|
|
|
|
let newIndex = 0;
|
|
|
|
let oldIndex = 0;
|
|
|
|
while (newIndex < newApps.length || oldIndex < oldApps.length) {
|
2017-06-12 21:41:42 -04:00
|
|
|
let oldApp = oldApps.length > oldIndex ? oldApps[oldIndex] : null;
|
|
|
|
let newApp = newApps.length > newIndex ? newApps[newIndex] : null;
|
|
|
|
|
2011-01-13 07:44:40 -05:00
|
|
|
// No change at oldIndex/newIndex
|
2017-06-12 21:41:42 -04:00
|
|
|
if (oldApp == newApp) {
|
2011-01-13 07:44:40 -05:00
|
|
|
oldIndex++;
|
|
|
|
newIndex++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// App removed at oldIndex
|
2018-07-14 16:56:22 -04:00
|
|
|
if (oldApp && !newApps.includes(oldApp)) {
|
2011-01-13 07:44:40 -05:00
|
|
|
removedActors.push(children[oldIndex]);
|
|
|
|
oldIndex++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// App added at newIndex
|
2018-07-14 16:56:22 -04:00
|
|
|
if (newApp && !oldApps.includes(newApp)) {
|
2017-06-12 21:41:42 -04:00
|
|
|
addedItems.push({ app: newApp,
|
|
|
|
item: this._createAppItem(newApp),
|
2011-01-13 07:44:40 -05:00
|
|
|
pos: newIndex });
|
|
|
|
newIndex++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// App moved
|
2019-08-19 15:33:15 -04:00
|
|
|
let nextApp = newApps.length > newIndex + 1
|
|
|
|
? newApps[newIndex + 1] : null;
|
2017-06-12 21:41:42 -04:00
|
|
|
let insertHere = nextApp && nextApp == oldApp;
|
2017-10-30 20:38:18 -04:00
|
|
|
let alreadyRemoved = removedActors.reduce((result, actor) => {
|
2012-12-22 08:57:06 -05:00
|
|
|
let removedApp = actor.child._delegate.app;
|
2017-06-12 21:41:42 -04:00
|
|
|
return result || removedApp == newApp;
|
2011-01-13 07:44:40 -05:00
|
|
|
}, false);
|
|
|
|
|
|
|
|
if (insertHere || alreadyRemoved) {
|
2017-06-12 21:41:42 -04:00
|
|
|
let newItem = this._createAppItem(newApp);
|
|
|
|
addedItems.push({ app: newApp,
|
2011-01-20 10:23:43 -05:00
|
|
|
item: newItem,
|
2011-01-13 07:44:40 -05:00
|
|
|
pos: newIndex + removedActors.length });
|
|
|
|
newIndex++;
|
|
|
|
} else {
|
|
|
|
removedActors.push(children[oldIndex]);
|
|
|
|
oldIndex++;
|
|
|
|
}
|
2010-07-21 19:29:02 -04:00
|
|
|
}
|
|
|
|
|
2019-08-19 20:51:42 -04:00
|
|
|
for (let i = 0; i < addedItems.length; i++) {
|
2012-12-22 08:57:06 -05:00
|
|
|
this._box.insert_child_at_index(addedItems[i].item,
|
2012-02-13 15:27:16 -05:00
|
|
|
addedItems[i].pos);
|
2019-08-19 20:51:42 -04:00
|
|
|
}
|
2011-01-13 07:44:40 -05:00
|
|
|
|
2011-09-19 17:47:16 -04:00
|
|
|
for (let i = 0; i < removedActors.length; i++) {
|
2012-12-22 08:57:06 -05:00
|
|
|
let item = removedActors[i];
|
2011-01-21 06:06:13 -05:00
|
|
|
|
2012-10-30 15:15:29 -04:00
|
|
|
// Don't animate item removal when the overview is transitioning
|
|
|
|
// or hidden
|
|
|
|
if (Main.overview.visible && !Main.overview.animationInProgress)
|
2011-09-19 17:47:16 -04:00
|
|
|
item.animateOutAndDestroy();
|
|
|
|
else
|
2012-04-08 15:35:41 -04:00
|
|
|
item.destroy();
|
2011-09-19 17:47:16 -04:00
|
|
|
}
|
2011-01-13 07:44:40 -05:00
|
|
|
|
2010-11-27 16:31:14 -05:00
|
|
|
this._adjustIconSize();
|
2011-01-21 06:06:13 -05:00
|
|
|
|
2011-02-02 10:44:07 -05:00
|
|
|
// Skip animations on first run when adding the initial set
|
|
|
|
// of items, to avoid all items zooming in at once
|
|
|
|
|
2012-12-22 09:26:12 -05:00
|
|
|
let animate = this._shownInitially && Main.overview.visible &&
|
|
|
|
!Main.overview.animationInProgress;
|
2011-02-06 11:46:12 -05:00
|
|
|
|
2012-12-22 09:26:12 -05:00
|
|
|
if (!this._shownInitially)
|
|
|
|
this._shownInitially = true;
|
|
|
|
|
2019-08-19 20:51:42 -04:00
|
|
|
for (let i = 0; i < addedItems.length; i++)
|
2012-12-22 09:26:12 -05:00
|
|
|
addedItems[i].item.show(animate);
|
2013-01-28 17:22:23 -05:00
|
|
|
|
2021-01-28 17:24:44 -05:00
|
|
|
// Update separator
|
|
|
|
const nFavorites = Object.keys(favorites).length;
|
2021-01-29 13:03:07 -05:00
|
|
|
const nIcons = children.length - removedActors.length;
|
|
|
|
if (nFavorites > 0 && nFavorites < nIcons) {
|
2021-01-28 17:24:44 -05:00
|
|
|
if (!this._separator) {
|
|
|
|
this._separator = new St.Widget({
|
|
|
|
style_class: 'dash-separator',
|
|
|
|
y_align: Clutter.ActorAlign.CENTER,
|
|
|
|
height: this.iconSize,
|
|
|
|
});
|
|
|
|
this._box.add_child(this._separator);
|
|
|
|
}
|
|
|
|
let pos = nFavorites;
|
|
|
|
if (this._dragPlaceholder)
|
|
|
|
pos++;
|
|
|
|
this._box.set_child_at_index(this._separator, pos);
|
|
|
|
} else if (this._separator) {
|
|
|
|
this._separator.destroy();
|
|
|
|
this._separator = null;
|
|
|
|
}
|
|
|
|
|
2013-01-28 17:22:23 -05:00
|
|
|
// Workaround for https://bugzilla.gnome.org/show_bug.cgi?id=692744
|
|
|
|
// Without it, StBoxLayout may use a stale size cache
|
|
|
|
this._box.queue_relayout();
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-07-21 19:29:02 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_clearDragPlaceholder() {
|
2010-11-07 20:51:02 -05:00
|
|
|
if (this._dragPlaceholder) {
|
2013-06-05 08:17:20 -04:00
|
|
|
this._animatingPlaceholdersCount++;
|
2011-01-21 06:06:13 -05:00
|
|
|
this._dragPlaceholder.animateOutAndDestroy();
|
2017-10-30 20:38:18 -04:00
|
|
|
this._dragPlaceholder.connect('destroy', () => {
|
|
|
|
this._animatingPlaceholdersCount--;
|
|
|
|
});
|
2010-11-07 20:51:02 -05:00
|
|
|
this._dragPlaceholder = null;
|
|
|
|
}
|
2013-05-31 11:47:48 -04:00
|
|
|
this._dragPlaceholderPos = -1;
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-11-07 20:51:02 -05:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_clearEmptyDropTarget() {
|
2013-05-31 11:59:27 -04:00
|
|
|
if (this._emptyDropTarget) {
|
|
|
|
this._emptyDropTarget.animateOutAndDestroy();
|
|
|
|
this._emptyDropTarget = null;
|
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2013-05-31 11:59:27 -04:00
|
|
|
|
2020-12-11 07:47:28 -05:00
|
|
|
handleDragOver(source, actor, x, _y, _time) {
|
2012-10-02 20:13:47 -04:00
|
|
|
let app = getAppFromSource(source);
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
// Don't allow favoriting of transient apps
|
Kill off ShellAppInfo, move into ShellApp
This dramatically thins down and sanitizes the application code.
The ShellAppSystem changes in a number of ways:
* Preferences are special cased more explicitly; they aren't apps,
they're shortcuts for an app), and we don't have many of them, so
don't need e.g. the optimizations in ShellAppSystem for searching.
* get_app() changes to lookup_app() and returns null if an app isn't
found. The semantics where it tried to find the .desktop file
if we didn't know about it were just broken; I am pretty sure no
caller needs this, and if they do we'll fix them.
* ShellAppSystem maintains two indexes on apps (by desktop file id
and by GMenuTreeEntry), but is no longer in the business of
dealing with GMenuTree as far as hierarchy and categories go. That
is moved up into js/ui/appDisplay.js. Actually, it flattens both
apps and settings.
Also, ShellWindowTracker is now the sole reference-owner for
window-backed apps. We still do the weird "window:0x1234beef" id
for these apps, but a reference is not stored in ShellAppSystem.
The js/ui/appDisplay.js code is rewritten, and sucks a lot less.
Variable names are clearer:
_apps -> _appIcons
_filterApp -> _visibleApps
_filters -> _categoryBox
Similarly for function names. We no longer call (for every app) a
recursive lookup in GMenuTree to see if it's in a particular section
on every category switch; it's all cached.
NOTE - this intentionally reverts the incremental loading code from
commit 7813c5b93f6bcde8c4beae286e82bfc472b2b656. It's fast enough
here without that.
https://bugzilla.gnome.org/show_bug.cgi?id=648149
2011-04-21 13:35:01 -04:00
|
|
|
if (app == null || app.is_window_backed())
|
2010-07-21 19:29:02 -04:00
|
|
|
return DND.DragMotionResult.NO_DROP;
|
|
|
|
|
2015-03-20 15:56:52 -04:00
|
|
|
if (!global.settings.is_writable('favorite-apps'))
|
2014-12-11 10:22:19 -05:00
|
|
|
return DND.DragMotionResult.NO_DROP;
|
|
|
|
|
2011-01-06 08:19:44 -05:00
|
|
|
let favorites = AppFavorites.getAppFavorites().getFavorites();
|
|
|
|
let numFavorites = favorites.length;
|
|
|
|
|
|
|
|
let favPos = favorites.indexOf(app);
|
|
|
|
|
2011-01-21 06:06:13 -05:00
|
|
|
let children = this._box.get_children();
|
|
|
|
let numChildren = children.length;
|
2020-12-11 07:47:28 -05:00
|
|
|
let boxWidth = this._box.width;
|
2010-11-07 20:51:02 -05:00
|
|
|
|
|
|
|
// Keep the placeholder out of the index calculation; assuming that
|
|
|
|
// the remove target has the same size as "normal" items, we don't
|
|
|
|
// need to do the same adjustment there.
|
|
|
|
if (this._dragPlaceholder) {
|
2020-12-11 07:47:28 -05:00
|
|
|
boxWidth -= this._dragPlaceholder.width;
|
2010-11-07 20:51:02 -05:00
|
|
|
numChildren--;
|
|
|
|
}
|
|
|
|
|
2021-01-28 17:24:44 -05:00
|
|
|
// Same with the separator
|
|
|
|
if (this._separator) {
|
|
|
|
boxWidth -= this._separator.width;
|
|
|
|
numChildren--;
|
|
|
|
}
|
|
|
|
|
2013-05-31 11:59:27 -04:00
|
|
|
let pos;
|
|
|
|
if (!this._emptyDropTarget)
|
2020-12-11 07:47:28 -05:00
|
|
|
pos = Math.floor(x * numChildren / boxWidth);
|
2013-05-31 11:59:27 -04:00
|
|
|
else
|
|
|
|
pos = 0; // always insert at the top when dash is empty
|
2012-01-26 09:35:31 -05:00
|
|
|
|
2020-05-13 12:28:53 -04:00
|
|
|
// Put the placeholder after the last favorite if we are not
|
|
|
|
// in the favorites zone
|
|
|
|
if (pos > numFavorites)
|
|
|
|
pos = numFavorites;
|
|
|
|
|
|
|
|
if (pos !== this._dragPlaceholderPos && this._animatingPlaceholdersCount === 0) {
|
2012-01-26 09:35:31 -05:00
|
|
|
this._dragPlaceholderPos = pos;
|
2011-01-06 08:19:44 -05:00
|
|
|
|
|
|
|
// Don't allow positioning before or after self
|
2011-01-21 06:06:13 -05:00
|
|
|
if (favPos != -1 && (pos == favPos || pos == favPos + 1)) {
|
2013-06-05 08:17:20 -04:00
|
|
|
this._clearDragPlaceholder();
|
2011-01-06 08:19:44 -05:00
|
|
|
return DND.DragMotionResult.CONTINUE;
|
2011-01-21 06:06:13 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the placeholder already exists, we just move
|
|
|
|
// it, but if we are adding it, expand its size in
|
|
|
|
// an animation
|
|
|
|
let fadeIn;
|
|
|
|
if (this._dragPlaceholder) {
|
2012-12-22 08:57:06 -05:00
|
|
|
this._dragPlaceholder.destroy();
|
2011-01-21 06:06:13 -05:00
|
|
|
fadeIn = false;
|
|
|
|
} else {
|
|
|
|
fadeIn = true;
|
|
|
|
}
|
2011-01-06 08:19:44 -05:00
|
|
|
|
2011-01-20 10:23:43 -05:00
|
|
|
this._dragPlaceholder = new DragPlaceholderItem();
|
2019-08-19 13:55:49 -04:00
|
|
|
this._dragPlaceholder.child.set_width(this.iconSize);
|
|
|
|
this._dragPlaceholder.child.set_height(this.iconSize / 2);
|
2012-12-22 08:57:06 -05:00
|
|
|
this._box.insert_child_at_index(this._dragPlaceholder,
|
2012-02-13 15:27:16 -05:00
|
|
|
this._dragPlaceholderPos);
|
2012-12-22 09:26:12 -05:00
|
|
|
this._dragPlaceholder.show(fadeIn);
|
2010-11-07 20:51:02 -05:00
|
|
|
}
|
|
|
|
|
2012-01-26 04:20:03 -05:00
|
|
|
if (!this._dragPlaceholder)
|
|
|
|
return DND.DragMotionResult.NO_DROP;
|
|
|
|
|
2019-08-19 15:38:51 -04:00
|
|
|
let srcIsFavorite = favPos != -1;
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
if (srcIsFavorite)
|
2010-11-07 20:51:02 -05:00
|
|
|
return DND.DragMotionResult.MOVE_DROP;
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
return DND.DragMotionResult.COPY_DROP;
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
// Draggable target interface
|
2019-01-31 09:08:10 -05:00
|
|
|
acceptDrop(source, _actor, _x, _y, _time) {
|
2012-10-02 20:13:47 -04:00
|
|
|
let app = getAppFromSource(source);
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
// Don't allow favoriting of transient apps
|
2019-08-19 20:51:42 -04:00
|
|
|
if (app == null || app.is_window_backed())
|
2010-07-21 19:29:02 -04:00
|
|
|
return false;
|
|
|
|
|
2015-03-20 15:56:52 -04:00
|
|
|
if (!global.settings.is_writable('favorite-apps'))
|
2014-12-11 10:22:19 -05:00
|
|
|
return false;
|
|
|
|
|
2010-07-21 19:29:02 -04:00
|
|
|
let id = app.get_id();
|
|
|
|
|
|
|
|
let favorites = AppFavorites.getAppFavorites().getFavoriteMap();
|
|
|
|
|
2019-08-19 15:38:51 -04:00
|
|
|
let srcIsFavorite = id in favorites;
|
2010-07-21 19:29:02 -04:00
|
|
|
|
2010-11-07 20:51:02 -05:00
|
|
|
let favPos = 0;
|
|
|
|
let children = this._box.get_children();
|
|
|
|
for (let i = 0; i < this._dragPlaceholderPos; i++) {
|
2011-01-21 06:06:13 -05:00
|
|
|
if (this._dragPlaceholder &&
|
2012-12-22 08:57:06 -05:00
|
|
|
children[i] == this._dragPlaceholder)
|
2011-01-21 06:06:13 -05:00
|
|
|
continue;
|
|
|
|
|
2012-12-22 08:57:06 -05:00
|
|
|
let childId = children[i].child._delegate.app.get_id();
|
2010-11-07 20:51:02 -05:00
|
|
|
if (childId == id)
|
|
|
|
continue;
|
|
|
|
if (childId in favorites)
|
|
|
|
favPos++;
|
|
|
|
}
|
|
|
|
|
2019-05-15 15:32:29 -04:00
|
|
|
// No drag placeholder means we don't want to favorite the app
|
2012-01-26 04:20:03 -05:00
|
|
|
// and we are dragging it to its original position
|
|
|
|
if (!this._dragPlaceholder)
|
|
|
|
return true;
|
|
|
|
|
2017-10-30 20:38:18 -04:00
|
|
|
Meta.later_add(Meta.LaterType.BEFORE_REDRAW, () => {
|
|
|
|
let appFavorites = AppFavorites.getAppFavorites();
|
|
|
|
if (srcIsFavorite)
|
|
|
|
appFavorites.moveFavoriteToPos(id, favPos);
|
|
|
|
else
|
|
|
|
appFavorites.addFavoriteAtPos(id, favPos);
|
|
|
|
return false;
|
|
|
|
});
|
2010-07-21 19:29:02 -04:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2019-07-16 05:24:13 -04:00
|
|
|
});
|