diff --git a/js/ui/overview.js b/js/ui/overview.js index 377c03445..41289f12a 100644 --- a/js/ui/overview.js +++ b/js/ui/overview.js @@ -159,6 +159,8 @@ const Overview = new Lang.Class({ Main.xdndHandler.connect('drag-begin', Lang.bind(this, this._onDragBegin)); Main.xdndHandler.connect('drag-end', Lang.bind(this, this._onDragEnd)); + global.screen.connect('restacked', Lang.bind(this, this._onRestacked)); + this._windowSwitchTimeoutId = 0; this._windowSwitchTimestamp = 0; this._lastActiveWorkspaceIndex = -1; @@ -217,6 +219,12 @@ const Overview = new Lang.Class({ this._dash.showAppsButton); this._group.add_actor(this._viewSelector.actor); + this._thumbnailsBox = new WorkspaceThumbnail.ThumbnailsBox(); + this._group.add_actor(this._thumbnailsBox.actor); + + // TODO: unique icon + Main.ctrlAltTabManager.addGroup(this._thumbnailsBox.actor, _("Workspaces"), 'user-bookmarks-symbolic'); + // Add our same-line elements after the search entry this._overview.add_actor(this._group); @@ -502,6 +510,18 @@ const Overview = new Lang.Class({ this._coverPane.set_size(primary.width, contentHeight); }, + _onRestacked: function() { + let stack = global.get_window_actors(); + let stackIndices = {}; + + for (let i = 0; i < stack.length; i++) { + // Use the stable sequence for an integer to use as a hash key + stackIndices[stack[i].get_meta_window().get_stable_sequence()] = i; + } + + this.emit('sync-window-stacking', stackIndices); + }, + //// Public methods //// beginAppDrag: function(source) { diff --git a/js/ui/workspaceThumbnail.js b/js/ui/workspaceThumbnail.js index 8fde2b9ef..58209ca1a 100644 --- a/js/ui/workspaceThumbnail.js +++ b/js/ui/workspaceThumbnail.js @@ -738,6 +738,9 @@ const ThumbnailsBox = new Lang.Class({ this._nWorkspacesNotifyId = global.screen.connect('notify::n-workspaces', Lang.bind(this, this._workspacesChanged)); + this._syncStackingId = + Main.overview.connect('sync-window-stacking', + Lang.bind(this, this.syncStacking)); this._targetScale = 0; this._scale = 0; @@ -770,6 +773,10 @@ const ThumbnailsBox = new Lang.Class({ global.screen.disconnect(this._nWorkspacesNotifyId); this._nWorkspacesNotifyId = 0; } + if (this._syncStackingId > 0) { + Main.overview.disconnect(this._syncStackingId); + this._syncStackingId = 0; + } for (let w = 0; w < this._thumbnails.length; w++) this._thumbnails[w].destroy(); @@ -882,7 +889,7 @@ const ThumbnailsBox = new Lang.Class({ this._queueUpdateStates(); }, - syncStacking: function(stackIndices) { + syncStacking: function(actor, stackIndices) { for (let i = 0; i < this._thumbnails.length; i++) this._thumbnails[i].syncStacking(stackIndices); }, diff --git a/js/ui/workspacesView.js b/js/ui/workspacesView.js index 039ad9b20..14a201180 100644 --- a/js/ui/workspacesView.js +++ b/js/ui/workspacesView.js @@ -22,8 +22,6 @@ const MAX_WORKSPACES = 16; const OVERRIDE_SCHEMA = 'org.gnome.shell.overrides'; -const CONTROLS_POP_IN_TIME = 0.1; - const WorkspacesView = new Lang.Class({ Name: 'WorkspacesView', @@ -460,23 +458,8 @@ const WorkspacesDisplay = new Lang.Class({ this.actor.connect('parent-set', Lang.bind(this, this._parentSet)); this.actor.set_clip_to_allocation(true); - let controls = new St.Bin({ style_class: 'workspace-controls', - request_mode: Clutter.RequestMode.WIDTH_FOR_HEIGHT, - y_align: St.Align.START, - y_fill: true }); - this._controls = controls; - this.actor.add_actor(controls); - - controls.reactive = true; - controls.track_hover = true; - controls.connect('notify::hover', - Lang.bind(this, this._onControlsHoverChanged)); - this._primaryIndex = Main.layoutManager.primaryIndex; - this._thumbnailsBox = new WorkspaceThumbnail.ThumbnailsBox(); - controls.add_actor(this._thumbnailsBox.actor); - this._workspacesViews = null; this._primaryScrollAdjustment = null; @@ -489,26 +472,6 @@ const WorkspacesDisplay = new Lang.Class({ this._inDrag = false; this._cancelledDrag = false; - this._controlsInitiallyHovered = false; - this._alwaysZoomOut = false; - this._zoomOut = false; - this._zoomFraction = 0; - - this._updateAlwaysZoom(); - - // If we stop hiding the overview on layout changes, we will need to - // update the _workspacesViews here - Main.layoutManager.connect('monitors-changed', Lang.bind(this, this._updateAlwaysZoom)); - - Main.xdndHandler.connect('drag-begin', Lang.bind(this, function(){ - this._alwaysZoomOut = true; - })); - - Main.xdndHandler.connect('drag-end', Lang.bind(this, function(){ - this._alwaysZoomOut = false; - this._updateAlwaysZoom(); - })); - global.screen.connect('notify::n-workspaces', Lang.bind(this, this._workspacesChanged)); @@ -536,31 +499,10 @@ const WorkspacesDisplay = new Lang.Class({ }, show: function() { - if(!this._alwaysZoomOut) { - let [mouseX, mouseY] = global.get_pointer(); - let [x, y] = this._controls.get_transformed_position(); - let [width, height] = this._controls.get_transformed_size(); - let visibleWidth = this._controls.get_theme_node().get_length('visible-width'); - let rtl = (Clutter.get_default_text_direction () == Clutter.TextDirection.RTL); - if(rtl) - x = x + width - visibleWidth; - if(mouseX > x - 0.5 && mouseX < x + visibleWidth + 0.5 && - mouseY > y - 0.5 && mouseY < y + height + 0.5) - this._controlsInitiallyHovered = true; - } - - this._zoomOut = this._alwaysZoomOut; - this._zoomFraction = this._alwaysZoomOut ? 1 : 0; - this._updateZoom(); - - this._controls.show(); - this._thumbnailsBox.show(); - this._updateSwitcherVisibility(); - this._updateWorkspacesViews(); this._restackedNotifyId = - global.screen.connect('restacked', + Main.overview.connect('sync-window-stacking', Lang.bind(this, this._onRestacked)); if (this._appDragBeginId == 0) @@ -581,8 +523,6 @@ const WorkspacesDisplay = new Lang.Class({ if (this._windowDragEndId == 0) this._windowDragEndId = Main.overview.connect('window-drag-end', Lang.bind(this, this._dragEnd)); - - this._onRestacked(); }, zoomFromOverview: function() { @@ -592,14 +532,8 @@ const WorkspacesDisplay = new Lang.Class({ }, hide: function() { - this._controls.hide(); - this._thumbnailsBox.hide(); - - if (!this._alwaysZoomOut) - this.zoomFraction = 0; - if (this._restackedNotifyId > 0){ - global.screen.disconnect(this._restackedNotifyId); + Main.overview.disconnect(this._restackedNotifyId); this._restackedNotifyId = 0; } if (this._appDragBeginId > 0) { @@ -743,73 +677,15 @@ const WorkspacesDisplay = new Lang.Class({ return this._getPrimaryView().getActiveWorkspace().hasMaximizedWindows(); }, - // zoomFraction property allows us to tween the controls sliding in and out - set zoomFraction(fraction) { - this._zoomFraction = fraction; - this.actor.queue_relayout(); - }, - - get zoomFraction() { - return this._zoomFraction; - }, - - _updateAlwaysZoom: function() { - // Always show the pager if workspaces are actually used, - // e.g. there are windows on more than one - this._alwaysZoomOut = global.screen.n_workspaces > 2; - - if (this._alwaysZoomOut) - return; - - let monitors = Main.layoutManager.monitors; - let primary = Main.layoutManager.primaryMonitor; - - /* Look for any monitor to the right of the primary, if there is - * one, we always keep zoom out, otherwise its hard to reach - * the thumbnail area without passing into the next monitor. */ - for (let i = 0; i < monitors.length; i++) { - if (monitors[i].x >= primary.x + primary.width) { - this._alwaysZoomOut = true; - break; - } - } - }, - _getPreferredWidth: function (actor, forHeight, alloc) { // pass through the call in case the child needs it, but report 0x0 - this._controls.get_preferred_width(forHeight); }, _getPreferredHeight: function (actor, forWidth, alloc) { // pass through the call in case the child needs it, but report 0x0 - this._controls.get_preferred_height(forWidth); }, _allocate: function (actor, box, flags) { - let childBox = new Clutter.ActorBox(); - - let totalWidth = box.x2 - box.x1; - - // width of the controls - let [controlsMin, controlsNatural] = this._controls.get_preferred_width(box.y2 - box.y1); - - // Amount of space on the screen we reserve for the visible control - let controlsVisible = this._controls.get_theme_node().get_length('visible-width'); - let controlsReserved = controlsVisible * (1 - this._zoomFraction) + controlsNatural * this._zoomFraction; - - let rtl = (Clutter.get_default_text_direction () == Clutter.TextDirection.RTL); - if (rtl) { - childBox.x2 = controlsReserved; - childBox.x1 = childBox.x2 - controlsNatural; - } else { - childBox.x1 = totalWidth - controlsReserved; - childBox.x2 = childBox.x1 + controlsNatural; - } - - childBox.y1 = 0; - childBox.y2 = box.y2- box.y1; - this._controls.allocate(childBox, flags); - this._updateWorkspacesGeometry(); }, @@ -849,24 +725,15 @@ const WorkspacesDisplay = new Lang.Class({ let width = fullWidth; let height = fullHeight; - let [controlsMin, controlsNatural] = this._controls.get_preferred_width(height); - let controlsVisible = this._controls.get_theme_node().get_length('visible-width'); - let [x, y] = this.actor.get_transformed_position(); let rtl = (Clutter.get_default_text_direction () == Clutter.TextDirection.RTL); - let clipWidth = width - controlsVisible; + let clipWidth = width; let clipHeight = fullHeight; - let clipX = rtl ? x + controlsVisible : x; + let clipX = x; let clipY = y + (fullHeight - clipHeight) / 2; - let widthAdjust = this._zoomOut ? controlsNatural : controlsVisible; - widthAdjust += Main.overview._spacing; - width -= widthAdjust; - if (rtl) - x += widthAdjust; - let monitors = Main.layoutManager.monitors; let m = 0; for (let i = 0; i < monitors.length; i++) { @@ -889,25 +756,12 @@ const WorkspacesDisplay = new Lang.Class({ } }, - _onRestacked: function() { - let stack = global.get_window_actors(); - let stackIndices = {}; - - for (let i = 0; i < stack.length; i++) { - // Use the stable sequence for an integer to use as a hash key - stackIndices[stack[i].get_meta_window().get_stable_sequence()] = i; - } - + _onRestacked: function(actor, stackIndices) { for (let i = 0; i < this._workspacesViews.length; i++) this._workspacesViews[i].syncStacking(stackIndices); - - this._thumbnailsBox.syncStacking(stackIndices); }, _workspacesChanged: function() { - this._updateAlwaysZoom(); - this._updateZoom(); - if (this._workspacesViews == null) return; @@ -962,35 +816,6 @@ const WorkspacesDisplay = new Lang.Class({ this._updateSwitcherVisibility(); }, - _updateZoom : function() { - if (Main.overview.animationInProgress) - return; - - let shouldZoom = this._alwaysZoomOut || this._controls.hover; - if (shouldZoom != this._zoomOut) { - this._zoomOut = shouldZoom; - this._updateWorkspacesGeometry(); - - if (!this._workspacesViews) - return; - - Tweener.addTween(this, - { zoomFraction: this._zoomOut ? 1 : 0, - time: WORKSPACE_SWITCH_TIME, - transition: 'easeOutQuad' }); - - for (let i = 0; i < this._workspacesViews.length; i++) - this._workspacesViews[i].updateWindowPositions(); - } - }, - - _onControlsHoverChanged: function() { - if(!this._controls.hover) - this._controlsInitiallyHovered = false; - if(!this._controlsInitiallyHovered) - this._updateZoom(); - }, - _dragBegin: function() { this._inDrag = true; this._cancelledDrag = false; @@ -1006,22 +831,11 @@ const WorkspacesDisplay = new Lang.Class({ }, _onDragMotion: function(dragEvent) { - let controlsHovered = this._controls.contains(dragEvent.targetActor); - this._controls.set_hover(controlsHovered); - return DND.DragMotionResult.CONTINUE; }, _dragEnd: function() { this._inDrag = false; - - // We do this deferred because drag-end is emitted before dnd.js emits - // event/leave events that were suppressed during the drag. If we didn't - // defer this, we'd zoom out then immediately zoom in because of the - // enter event we received. That would normally be invisible but we - // might as well avoid it. - Meta.later_add(Meta.LaterType.BEFORE_REDRAW, - Lang.bind(this, this._updateZoom)); } }); Signals.addSignalMethods(WorkspacesDisplay.prototype);