From e7fcce3484bff036d960f20fd3a411d1cd700dde Mon Sep 17 00:00:00 2001 From: Tanner Doshier Date: Fri, 10 Aug 2012 13:13:39 -0500 Subject: [PATCH] overview, workspacesView: Use ThumbnailsBox for independent workspace thumbnails Both WorkspacesDisplay and ThumbnailsBox need to know when windows have been restacked. Instead of each tracking changes on their own or trying to call each other, have the overview keep track and do the calculations, emitting a signal with the result. https://bugzilla.gnome.org/show_bug.cgi?id=682050 --- js/ui/overview.js | 20 ++++ js/ui/workspaceThumbnail.js | 9 +- js/ui/workspacesView.js | 196 +----------------------------------- 3 files changed, 33 insertions(+), 192 deletions(-) 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);