diff options
Diffstat (limited to 'extensions/46/vertical-workspaces/lib')
26 files changed, 0 insertions, 13518 deletions
diff --git a/extensions/46/vertical-workspaces/lib/appDisplay.js b/extensions/46/vertical-workspaces/lib/appDisplay.js deleted file mode 100644 index d94f7df..0000000 --- a/extensions/46/vertical-workspaces/lib/appDisplay.js +++ /dev/null @@ -1,2014 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * appDisplay.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import Gio from 'gi://Gio'; -import GLib from 'gi://GLib'; -import GObject from 'gi://GObject'; -import Graphene from 'gi://Graphene'; -import Meta from 'gi://Meta'; -import Pango from 'gi://Pango'; -import Shell from 'gi://Shell'; -import St from 'gi://St'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as AppDisplay from 'resource:///org/gnome/shell/ui/appDisplay.js'; -import * as DND from 'resource:///org/gnome/shell/ui/dnd.js'; -import * as PageIndicators from 'resource:///org/gnome/shell/ui/pageIndicators.js'; - -import { IconSize } from './iconGrid.js'; - -let Me; -let opt; -// gettext -let _; - -let _appDisplay; -let _timeouts; - -const APP_ICON_TITLE_EXPAND_TIME = 200; -const APP_ICON_TITLE_COLLAPSE_TIME = 100; - -const shellVersion46 = !Clutter.Container; // Container has been removed in 46 - -function _getCategories(info) { - let categoriesStr = info.get_categories(); - if (!categoriesStr) - return []; - return categoriesStr.split(';'); -} - -function _listsIntersect(a, b) { - for (let itemA of a) { - if (b.includes(itemA)) - return true; - } - return false; -} - -export const AppDisplayModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - _ = Me.gettext; - - _appDisplay = Main.overview._overview.controls._appDisplay; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - - this._appSystemStateConId = 0; - this._appGridLayoutConId = 0; - this._origAppViewItemAcceptDrop = null; - this._updateFolderIcons = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - _ = null; - _appDisplay = null; - } - - update(reset) { - this._removeTimeouts(); - this.moduleEnabled = opt.get('appDisplayModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - this.moduleEnabled = false; - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) { - this.moduleEnabled = false; - console.debug(' AppDisplayModule - Keeping untouched'); - } - } - - _activateModule() { - Me.Modules.iconGridModule.update(); - - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - _timeouts = {}; - - this._applyOverrides(); - this._updateAppDisplay(); - _appDisplay.add_style_class_name('app-display-46'); - - console.debug(' AppDisplayModule - Activated'); - } - - _disableModule() { - Me.Modules.iconGridModule.update(true); - - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - const reset = true; - this._updateAppDisplay(reset); - this._restoreOverviewGroup(); - - _appDisplay.remove_style_class_name('app-display-46'); - - console.debug(' AppDisplayModule - Disabled'); - } - - _removeTimeouts() { - if (_timeouts) { - Object.values(_timeouts).forEach(t => { - if (t) - GLib.source_remove(t); - }); - _timeouts = null; - } - } - - _applyOverrides() { - // Common/appDisplay - // this._overrides.addOverride('BaseAppViewCommon', AppDisplay.BaseAppView.prototype, BaseAppViewCommon); - // instead of overriding inaccessible BaseAppView class, we override its subclasses - AppDisplay and FolderView - this._overrides.addOverride('BaseAppViewCommonApp', AppDisplay.AppDisplay.prototype, BaseAppViewCommon); - this._overrides.addOverride('AppDisplay', AppDisplay.AppDisplay.prototype, AppDisplayCommon); - this._overrides.addOverride('AppViewItem', AppDisplay.AppViewItem.prototype, AppViewItemCommon); - this._overrides.addOverride('AppGridCommon', AppDisplay.AppGrid.prototype, AppGridCommon); - this._overrides.addOverride('AppIcon', AppDisplay.AppIcon.prototype, AppIcon); - if (opt.ORIENTATION) { - this._overrides.removeOverride('AppGridLayoutHorizontal'); - this._overrides.addOverride('AppGridLayoutVertical', _appDisplay._appGridLayout, BaseAppViewGridLayoutVertical); - } else { - this._overrides.removeOverride('AppGridLayoutVertical'); - this._overrides.addOverride('AppGridLayoutHorizontal', _appDisplay._appGridLayout, BaseAppViewGridLayoutHorizontal); - } - - // Custom folders - this._overrides.addOverride('BaseAppViewCommonFolder', AppDisplay.FolderView.prototype, BaseAppViewCommon); - this._overrides.addOverride('FolderView', AppDisplay.FolderView.prototype, FolderView); - this._overrides.addOverride('AppFolderDialog', AppDisplay.AppFolderDialog.prototype, AppFolderDialog); - this._overrides.addOverride('FolderIcon', AppDisplay.FolderIcon.prototype, FolderIcon); - - // Prevent changing grid page size when showing/hiding _pageIndicators - this._overrides.addOverride('PageIndicators', PageIndicators.PageIndicators.prototype, PageIndicatorsCommon); - } - - _updateAppDisplay(reset) { - const orientation = reset ? Clutter.Orientation.HORIZONTAL : opt.ORIENTATION; - BaseAppViewCommon._adaptForOrientation.bind(_appDisplay)(orientation); - - this._updateFavoritesConnection(reset); - - _appDisplay.visible = true; - if (reset) { - _appDisplay._grid.layoutManager.fixedIconSize = -1; - _appDisplay._grid.layoutManager.allow_incomplete_pages = true; - _appDisplay._grid._currentMode = -1; - _appDisplay._grid.setGridModes(); - _appDisplay._grid.set_style(''); - _appDisplay._prevPageArrow.set_scale(1, 1); - _appDisplay._nextPageArrow.set_scale(1, 1); - if (this._appGridLayoutConId) { - global.settings.disconnect(this._appGridLayoutConId); - this._appGridLayoutConId = 0; - } - this._repopulateAppDisplay(reset); - } else { - _appDisplay._grid._currentMode = -1; - // update grid on layout reset - if (!this._appGridLayoutConId) - this._appGridLayoutConId = global.settings.connect('changed::app-picker-layout', this._updateLayout.bind(this)); - - // avoid resetting appDisplay before startup animation - // x11 shell restart skips startup animation - if (!Main.layoutManager._startingUp) { - this._repopulateAppDisplay(); - } else if (Main.layoutManager._startingUp && Meta.is_restart()) { - _timeouts.three = GLib.idle_add(GLib.PRIORITY_LOW, () => { - this._repopulateAppDisplay(); - _timeouts.three = 0; - return GLib.SOURCE_REMOVE; - }); - } - } - } - - _updateFavoritesConnection(reset) { - if (!reset) { - if (!this._appSystemStateConId && opt.APP_GRID_INCLUDE_DASH >= 3) { - this._appSystemStateConId = Shell.AppSystem.get_default().connect( - 'app-state-changed', - () => { - this._updateFolderIcons = true; - _appDisplay._redisplay(); - } - ); - } - } else if (this._appSystemStateConId) { - Shell.AppSystem.get_default().disconnect(this._appSystemStateConId); - this._appSystemStateConId = 0; - } - } - - _restoreOverviewGroup() { - Main.overview.dash.showAppsButton.checked = false; - Main.layoutManager.overviewGroup.opacity = 255; - Main.layoutManager.overviewGroup.scale_x = 1; - Main.layoutManager.overviewGroup.scale_y = 1; - Main.layoutManager.overviewGroup.hide(); - _appDisplay.translation_x = 0; - _appDisplay.translation_y = 0; - _appDisplay.visible = true; - _appDisplay.opacity = 255; - } - - _updateLayout(settings, key) { - // Reset the app grid only if the user layout has been completely removed - if (!settings.get_value(key).deep_unpack().length) { - this._repopulateAppDisplay(); - } - } - - _repopulateAppDisplay(reset = false, callback) { - // Remove all icons so they can be re-created with the current configuration - // Updating appGrid content while rebasing extensions when session is locked makes no sense (relevant for GS version < 46) - if (!Main.sessionMode.isLocked) - AppDisplayCommon.removeAllItems.bind(_appDisplay)(); - - // appDisplay disabled - if (reset) { - _appDisplay._redisplay(); - return; - } - - _appDisplay._readyToRedisplay = true; - _appDisplay._redisplay(); - - // Setting OffscreenRedirect should improve performance when opacity transitions are used - _appDisplay.offscreen_redirect = Clutter.OffscreenRedirect.ALWAYS; - - if (opt.APP_GRID_PERFORMANCE) - this._realizeAppDisplay(callback); - else if (callback) - callback(); - } - - _realizeAppDisplay(callback) { - // Workaround - silently realize appDisplay - // The realization takes some time and affects animations during the first use - // If we do it invisibly before the user needs the app grid, it can improve the user's experience - _appDisplay.opacity = 1; - - this._exposeAppGrid(); - _appDisplay._redisplay(); - this._exposeAppFolders(); - - // Let the main loop process our changes before we continue - _timeouts.updateAppGrid = GLib.idle_add(GLib.PRIORITY_LOW, () => { - this._restoreAppGrid(); - Me._resetInProgress = false; - - if (callback) - callback(); - - _timeouts.updateAppGrid = 0; - return GLib.SOURCE_REMOVE; - }); - } - - _exposeAppGrid() { - const overviewGroup = Main.layoutManager.overviewGroup; - if (!overviewGroup.visible) { - // scale down the overviewGroup so it don't cover uiGroup - overviewGroup.scale_y = 0.001; - // make it invisible to the eye, but visible for the renderer - overviewGroup.opacity = 1; - // if overview is hidden, show it - overviewGroup.visible = true; - } - } - - _restoreAppGrid() { - if (opt.APP_GRID_PERFORMANCE) - this._hideAppFolders(); - - const overviewGroup = Main.layoutManager.overviewGroup; - if (!Main.overview._shown) - overviewGroup.hide(); - overviewGroup.scale_y = 1; - overviewGroup.opacity = 255; - _appDisplay.opacity = 0; - _appDisplay.visible = false; - } - - _exposeAppFolders() { - _appDisplay._folderIcons.forEach(d => { - d._ensureFolderDialog(); - d._dialog.scale_y = 0.0001; - d._dialog.show(); - d._dialog._updateFolderSize(); - }); - } - - _hideAppFolders() { - _appDisplay._folderIcons.forEach(d => { - if (d._dialog) { - d._dialog.hide(); - d._dialog.scale_y = 1; - } - }); - } -}; - -function _getViewFromIcon(icon) { - icon = icon._sourceItem ? icon._sourceItem : icon; - for (let parent = icon.get_parent(); parent; parent = parent.get_parent()) { - if (parent instanceof AppDisplay.AppDisplay || parent instanceof AppDisplay.FolderView) { - return parent; - } - } - return null; -} - -const AppDisplayCommon = { - _ensureDefaultFolders() { - // disable creation of default folders if user deleted them - }, - - removeAllItems() { - this._orderedItems.slice().forEach(item => { - if (item._dialog) - Main.layoutManager.overviewGroup.remove_child(item._dialog); - - this._removeItem(item); - item.destroy(); - }); - this._folderIcons = []; - }, - - // apps load adapted for custom sorting and including dash items - _loadApps() { - let appIcons = []; - const runningApps = Shell.AppSystem.get_default().get_running().map(a => a.id); - - this._appInfoList = Shell.AppSystem.get_default().get_installed().filter(appInfo => { - try { - appInfo.get_id(); // catch invalid file encodings - } catch (e) { - return false; - } - - const appIsRunning = runningApps.includes(appInfo.get_id()); - const appIsFavorite = this._appFavorites.isFavorite(appInfo.get_id()); - const excludeApp = (opt.APP_GRID_EXCLUDE_RUNNING && appIsRunning) || (opt.APP_GRID_EXCLUDE_FAVORITES && appIsFavorite); - - return this._parentalControlsManager.shouldShowApp(appInfo) && !excludeApp; - }); - - let apps = this._appInfoList.map(app => app.get_id()); - - let appSys = Shell.AppSystem.get_default(); - - const appsInsideFolders = new Set(); - this._folderIcons = []; - if (!opt.APP_GRID_USAGE) { - let folders = this._folderSettings.get_strv('folder-children'); - folders.forEach(id => { - let path = `${this._folderSettings.path}folders/${id}/`; - let icon = this._items.get(id); - if (!icon) { - icon = new AppDisplay.FolderIcon(id, path, this); - icon.connect('apps-changed', () => { - this._redisplay(); - this._savePages(); - }); - icon.connect('notify::pressed', () => { - if (icon.pressed) - this.updateDragFocus(icon); - }); - } else if (this._updateFolderIcons && opt.APP_GRID_EXCLUDE_RUNNING) { - // if any app changed its running state, update folder icon - icon.icon.update(); - } - - // remove empty folder icons - if (!icon.visible) { - icon.destroy(); - return; - } - - appIcons.push(icon); - this._folderIcons.push(icon); - - icon.getAppIds().forEach(appId => appsInsideFolders.add(appId)); - }); - } - - // reset request to update active icon - this._updateFolderIcons = false; - - // Allow dragging of the icon only if the Dash would accept a drop to - // change favorite-apps. There are no other possible drop targets from - // the app picker, so there's no other need for a drag to start, - // at least on single-monitor setups. - // This also disables drag-to-launch on multi-monitor setups, - // but we hope that is not used much. - const isDraggable = - global.settings.is_writable('favorite-apps') || - global.settings.is_writable('app-picker-layout'); - - apps.forEach(appId => { - if (!opt.APP_GRID_USAGE && appsInsideFolders.has(appId)) - return; - - let icon = this._items.get(appId); - if (!icon) { - let app = appSys.lookup_app(appId); - icon = new AppDisplay.AppIcon(app, { isDraggable }); - icon.connect('notify::pressed', () => { - if (icon.pressed) - this.updateDragFocus(icon); - }); - } - - appIcons.push(icon); - }); - - // At last, if there's a placeholder available, add it - if (this._placeholder) - appIcons.push(this._placeholder); - - return appIcons; - }, - - _onDragBegin(overview, source) { - // let sourceId; - // support active preview icons - if (source._sourceItem) { - // sourceId = source._sourceFolder._id; - source = source._sourceItem; - } /* else { - sourceId = source.id; - }*/ - // Prevent switching page when an item on another page is selected - // by removing the focus from all icons - // This is an upstream bug - // this.selectApp(sourceId); - this.grab_key_focus(); - - this._dragMonitor = { - dragMotion: this._onDragMotion.bind(this), - dragDrop: this._onDragDrop.bind(this), - }; - DND.addDragMonitor(this._dragMonitor); - - this._appGridLayout.showPageIndicators(); - this._dragFocus = null; - this._swipeTracker.enabled = false; - - // When dragging from a folder dialog, the dragged app icon doesn't - // exist in AppDisplay. We work around that by adding a placeholder - // icon that is either destroyed on cancel, or becomes the effective - // new icon when dropped. - if (/* AppDisplay.*/_getViewFromIcon(source) instanceof AppDisplay.FolderView || - (opt.APP_GRID_EXCLUDE_FAVORITES && this._appFavorites.isFavorite(source.id))) - this._ensurePlaceholder(source); - }, - - _ensurePlaceholder(source) { - if (this._placeholder) - return; - - if (source._sourceItem) - source = source._sourceItem; - - const appSys = Shell.AppSystem.get_default(); - const app = appSys.lookup_app(source.id); - - const isDraggable = - global.settings.is_writable('favorite-apps') || - global.settings.is_writable('app-picker-layout'); - - this._placeholder = new AppDisplay.AppIcon(app, { isDraggable }); - this._placeholder.connect('notify::pressed', () => { - if (this._placeholder?.pressed) - this.updateDragFocus(this._placeholder); - }); - this._placeholder.scaleAndFade(); - this._redisplay(); - }, - - // accept source from active folder preview - acceptDrop(source) { - if (opt.APP_GRID_USAGE) - return false; - if (source._sourceItem) - source = source._sourceItem; - if (!this._acceptDropCommon(source)) - return false; - - this._savePages(); - - const view = /* AppDisplay.*/_getViewFromIcon(source); - if (view instanceof AppDisplay.FolderView) - view.removeApp(source.app); - - if (this._currentDialog) - this._currentDialog.popdown(); - - if (opt.APP_GRID_EXCLUDE_FAVORITES && this._appFavorites.isFavorite(source.id)) - this._appFavorites.removeFavorite(source.id); - return true; - }, - - _savePages() { - // Skip saving pages when search app grid mode is active - // and the grid is showing search results - if (Main.overview._overview.controls._origAppGridContent) - return; - - const pages = []; - - for (let i = 0; i < this._grid.nPages; i++) { - const pageItems = - this._grid.getItemsAtPage(i).filter(c => c.visible); - const pageData = {}; - - pageItems.forEach((item, index) => { - pageData[item.id] = { - position: GLib.Variant.new_int32(index), - }; - }); - pages.push(pageData); - } - - this._pageManager.pages = pages; - }, -}; - -const BaseAppViewCommon = { - after__init() { - // Only folders can run this init - this._isFolder = true; - - this._adaptForOrientation(opt.ORIENTATION, true); - - // Because the original class prototype is not exported, we need to inject every instance - const overrides = new Me.Util.Overrides(); - if (opt.ORIENTATION) { - overrides.addOverride('FolderGridLayoutVertical', this._appGridLayout, BaseAppViewGridLayoutVertical); - this._pageIndicators.set_style('margin-right: 12px;'); - } else { - overrides.addOverride('FolderGridLayoutHorizontal', this._appGridLayout, BaseAppViewGridLayoutHorizontal); - this._pageIndicators.set_style('margin-bottom: 12px;'); - } - }, - - _adaptForOrientation(orientation, folder) { - const vertical = !!orientation; - - this._grid.layoutManager.fixedIconSize = folder ? opt.APP_GRID_FOLDER_ICON_SIZE : opt.APP_GRID_ICON_SIZE; - this._grid.layoutManager._orientation = orientation; - this._orientation = orientation; - this._swipeTracker.orientation = orientation; - this._swipeTracker._reset(); - - this._adjustment = vertical - ? this._scrollView.get_vscroll_bar().adjustment - : this._scrollView.get_hscroll_bar().adjustment; - - this._prevPageArrow.pivot_point = new Graphene.Point({ x: 0.5, y: 0.5 }); - this._prevPageArrow.rotation_angle_z = vertical ? 90 : 0; - - this._nextPageArrow.pivot_point = new Graphene.Point({ x: 0.5, y: 0.5 }); - this._nextPageArrow.rotation_angle_z = vertical ? 90 : 0; - - const pageIndicators = this._pageIndicators; - pageIndicators.vertical = vertical; - this._box.vertical = !vertical; - pageIndicators.x_expand = !vertical; - pageIndicators.y_align = vertical ? Clutter.ActorAlign.CENTER : Clutter.ActorAlign.START; - pageIndicators.x_align = vertical ? Clutter.ActorAlign.START : Clutter.ActorAlign.CENTER; - - this._grid.layoutManager.allow_incomplete_pages = folder ? false : opt.APP_GRID_ALLOW_INCOMPLETE_PAGES; - const spacing = folder ? opt.APP_GRID_FOLDER_SPACING : opt.APP_GRID_SPACING; - this._grid.set_style(`column-spacing: ${spacing}px; row-spacing: ${spacing}px;`); - - if (vertical) { - this._scrollView.set_policy(St.PolicyType.NEVER, St.PolicyType.EXTERNAL); - if (!this._scrollConId) { - this._scrollConId = this._adjustment.connect('notify::value', adj => { - const value = adj.value / adj.page_size; - this._pageIndicators.setCurrentPosition(value); - }); - } - pageIndicators.remove_style_class_name('page-indicators-horizontal'); - pageIndicators.add_style_class_name('page-indicators-vertical'); - this._prevPageIndicator.add_style_class_name('prev-page-indicator'); - this._nextPageIndicator.add_style_class_name('next-page-indicator'); - this._nextPageArrow.translationY = 0; - this._prevPageArrow.translationY = 0; - this._nextPageIndicator.translationX = 0; - this._prevPageIndicator.translationX = 0; - } else { - this._scrollView.set_policy(St.PolicyType.EXTERNAL, St.PolicyType.NEVER); - if (this._scrollConId) { - this._adjustment.disconnect(this._scrollConId); - this._scrollConId = 0; - } - pageIndicators.remove_style_class_name('page-indicators-vertical'); - pageIndicators.add_style_class_name('page-indicators-horizontal'); - this._prevPageIndicator.remove_style_class_name('prev-page-indicator'); - this._nextPageIndicator.remove_style_class_name('next-page-indicator'); - this._nextPageArrow.translationX = 0; - this._prevPageArrow.translationX = 0; - this._nextPageIndicator.translationY = 0; - this._prevPageIndicator.translationY = 0; - } - - const scale = opt.APP_GRID_SHOW_PAGE_ARROWS ? 1 : 0; - this._prevPageArrow.set_scale(scale, scale); - this._nextPageArrow.set_scale(scale, scale); - }, - - _sortItemsByName(items) { - items.sort((a, b) => a.name.toLowerCase().localeCompare(b.name.toLowerCase())); - }, - - _updateItemPositions(icons, allowIncompletePages = false) { - // Avoid recursion when relocating icons - this._grid.layoutManager._skipRelocateSurplusItems = true; - - const { itemsPerPage } = this._grid; - - icons.slice().forEach((icon, index) => { - const [currentPage, currentPosition] = this._grid.layoutManager.getItemPosition(icon); - - let page, position; - if (allowIncompletePages) { - [page, position] = this._getItemPosition(icon); - } else { - page = Math.floor(index / itemsPerPage); - position = index % itemsPerPage; - } - - if (currentPage !== page || currentPosition !== position) { - this._moveItem(icon, page, position); - } - }); - - this._grid.layoutManager._skipRelocateSurplusItems = false; - // Disable animating the icons to their new positions - // since it can cause glitches when the app grid search mode is active - // and many icons are repositioning at once - this._grid.layoutManager._shouldEaseItems = false; - }, - - // Adds sorting options - _redisplay() { - // different options for main app grid and app folders - const thisIsFolder = this instanceof AppDisplay.FolderView; - const thisIsAppDisplay = !thisIsFolder; - - // When an app was dragged from a folder and dropped to the main grid - // folders (if exist) need to be redisplayed even if we temporary block it for the appDisplay - this._folderIcons?.forEach(icon => { - icon.view._redisplay(); - }); - - // Avoid unwanted updates - if (thisIsAppDisplay && !this._readyToRedisplay) - return; - - const oldApps = this._orderedItems.slice(); - const oldAppIds = oldApps.map(icon => icon.id); - - const newApps = this._loadApps(); - const newAppIds = newApps.map(icon => icon.id); - - const addedApps = newApps.filter(icon => !oldAppIds.includes(icon.id)); - const removedApps = oldApps.filter(icon => !newAppIds.includes(icon.id)); - - // Don't update folder without dialog if its content didn't change - if (!addedApps.length && !removedApps.length && thisIsFolder && !this.get_parent()) - return; - - // Remove old app icons - removedApps.forEach(icon => { - this._removeItem(icon); - icon.destroy(); - }); - - // For the main app grid only - let allowIncompletePages = thisIsAppDisplay && opt.APP_GRID_ALLOW_INCOMPLETE_PAGES; - - const customOrder = !((opt.APP_GRID_ORDER && thisIsAppDisplay) || (opt.APP_FOLDER_ORDER && thisIsFolder)); - if (!customOrder) { - allowIncompletePages = false; - - // Sort by name - this._sortItemsByName(newApps); - - // Sort by usage - if ((opt.APP_GRID_USAGE && thisIsAppDisplay) || - (opt.APP_FOLDER_USAGE && thisIsFolder)) { - newApps.sort((a, b) => Shell.AppUsage.get_default().compare(a.app?.id, b.app?.id)); - } - - // Sort favorites first - if (!opt.APP_GRID_EXCLUDE_FAVORITES && opt.APP_GRID_DASH_FIRST) { - const fav = Object.keys(this._appFavorites._favorites); - newApps.sort((a, b) => { - let aFav = fav.indexOf(a.id); - if (aFav < 0) - aFav = 999; - let bFav = fav.indexOf(b.id); - if (bFav < 0) - bFav = 999; - return bFav < aFav; - }); - } - - // Sort running first - if (!opt.APP_GRID_EXCLUDE_RUNNING && opt.APP_GRID_DASH_FIRST) { - newApps.sort((a, b) => a.app?.get_state() !== Shell.AppState.RUNNING && b.app?.get_state() === Shell.AppState.RUNNING); - } - - // Sort folders first - if (thisIsAppDisplay && opt.APP_GRID_FOLDERS_FIRST) - newApps.sort((a, b) => b._folder && !a._folder); - - // Sort folders last - else if (thisIsAppDisplay && opt.APP_GRID_FOLDERS_LAST) - newApps.sort((a, b) => a._folder && !b._folder); - } else { - // Sort items according to the custom order stored in pageManager - newApps.sort(this._compareItems.bind(this)); - } - - // Add new app icons to the grid - newApps.forEach(icon => { - const [page, position] = this._grid.getItemPosition(icon); - if (page === -1 && position === -1) - this._addItem(icon, -1, -1); - }); - // When a placeholder icon was added to the custom sorted grid during DND from a folder - // update its initial position on the page - if (customOrder) - newApps.sort(this._compareItems.bind(this)); - - this._orderedItems = newApps; - - // Update icon positions if needed - this._updateItemPositions(this._orderedItems, allowIncompletePages); - - // Relocate items with invalid positions - if (thisIsAppDisplay) { - const nPages = this._grid.layoutManager.nPages; - for (let pageIndex = 0; pageIndex < nPages; pageIndex++) - this._grid.layoutManager._relocateSurplusItems(pageIndex); - } - - this.emit('view-loaded'); - }, - - _canAccept(source) { - return source instanceof AppDisplay.AppViewItem; - }, - - // this method is replacing BaseAppVew.acceptDrop which can't be overridden directly - _acceptDropCommon(source) { - const dropTarget = this._dropTarget; - delete this._dropTarget; - if (!this._canAccept(source)) - return false; - - if (dropTarget === this._prevPageIndicator || - dropTarget === this._nextPageIndicator) { - let increment; - increment = dropTarget === this._prevPageIndicator ? -1 : 1; - const { currentPage, nPages } = this._grid; - const page = Math.min(currentPage + increment, nPages); - const position = page < nPages ? -1 : 0; - - this._moveItem(source, page, position); - this.goToPage(page); - } else if (this._delayedMoveData) { - // Dropped before the icon was moved - const { page, position } = this._delayedMoveData; - try { - this._moveItem(source, page, position); - } catch (e) { - console.warn(`Warning:${e}`); - } - this._removeDelayedMove(); - } - - return true; - }, - - // support active preview icons - _onDragMotion(dragEvent) { - if (!(dragEvent.source instanceof AppDisplay.AppViewItem)) - return DND.DragMotionResult.CONTINUE; - - if (dragEvent.source._sourceItem) - dragEvent.source = dragEvent.source._sourceItem; - - const appIcon = dragEvent.source; - - if (appIcon instanceof AppDisplay.AppViewItem) { - if (!this._dragMaybeSwitchPageImmediately(dragEvent)) { - // Two ways of switching pages during DND: - // 1) When "bumping" the cursor against the monitor edge, we switch - // page immediately. - // 2) When hovering over the next-page indicator for a certain time, - // we also switch page. - - const { targetActor } = dragEvent; - - if (targetActor === this._prevPageIndicator || - targetActor === this._nextPageIndicator) - this._maybeSetupDragPageSwitchInitialTimeout(dragEvent); - else - this._resetDragPageSwitch(); - } - } - - const thisIsFolder = this instanceof AppDisplay.FolderView; - const thisIsAppDisplay = !thisIsFolder; - - // Prevent reorganizing the main app grid icons when an app folder is open and when sorting is not custom - // For some reason in V-Shell the drag motion events propagate from folder to main grid, which is not a problem in default code - so test the open dialog - if (!this._currentDialog && (!opt.APP_GRID_ORDER && thisIsAppDisplay) || (!opt.APP_FOLDER_ORDER && thisIsFolder)) - this._maybeMoveItem(dragEvent); - - return DND.DragMotionResult.CONTINUE; - }, -}; - -const BaseAppViewGridLayoutHorizontal = { - _getIndicatorsWidth(box) { - const [width, height] = box.get_size(); - const arrows = [ - this._nextPageArrow, - this._previousPageArrow, - ]; - - let minArrowsWidth; - - minArrowsWidth = arrows.reduce( - (previousWidth, accessory) => { - const [min] = accessory.get_preferred_width(height); - return Math.max(previousWidth, min); - }, 0); - - minArrowsWidth = opt.APP_GRID_SHOW_PAGE_ARROWS ? minArrowsWidth : 0; - - const indicatorWidth = !this._grid._isFolder - ? minArrowsWidth + ((width - minArrowsWidth) * (1 - opt.APP_GRID_PAGE_WIDTH_SCALE)) / 2 - : minArrowsWidth + 6; - - return Math.round(indicatorWidth); - }, - - vfunc_allocate(container, box) { - const ltr = container.get_text_direction() !== Clutter.TextDirection.RTL; - const indicatorsWidth = this._getIndicatorsWidth(box); - - const pageIndicatorsHeight = 20; // _appDisplay._pageIndicators.height is unstable, 20 is determined by the style - const availHeight = box.get_height() - pageIndicatorsHeight; - const vPadding = Math.round((availHeight - availHeight * opt.APP_GRID_PAGE_HEIGHT_SCALE) / 2); - this._grid.indicatorsPadding = new Clutter.Margin({ - left: indicatorsWidth, - right: indicatorsWidth, - top: vPadding + pageIndicatorsHeight, - bottom: vPadding, - }); - - this._scrollView.allocate(box); - - const leftBox = box.copy(); - leftBox.x2 = leftBox.x1 + indicatorsWidth; - - const rightBox = box.copy(); - rightBox.x1 = rightBox.x2 - indicatorsWidth; - - this._previousPageIndicator.allocate(ltr ? leftBox : rightBox); - this._previousPageArrow.allocate_align_fill(ltr ? leftBox : rightBox, - 0.5, 0.5, false, false); - this._nextPageIndicator.allocate(ltr ? rightBox : leftBox); - this._nextPageArrow.allocate_align_fill(ltr ? rightBox : leftBox, - 0.5, 0.5, false, false); - - this._pageWidth = box.get_width(); - - // Center page arrow buttons - this._previousPageArrow.translationY = pageIndicatorsHeight / 2; - this._nextPageArrow.translationY = pageIndicatorsHeight / 2; - // Reset page indicators vertical position - this._nextPageIndicator.translationY = 0; - this._previousPageIndicator.translationY = 0; - }, -}; - -const BaseAppViewGridLayoutVertical = { - _getIndicatorsHeight(box) { - const [width, height] = box.get_size(); - const arrows = [ - this._nextPageArrow, - this._previousPageArrow, - ]; - - let minArrowsHeight; - - minArrowsHeight = arrows.reduce( - (previousHeight, accessory) => { - const [min] = accessory.get_preferred_height(width); - return Math.max(previousHeight, min); - }, 0); - - minArrowsHeight = opt.APP_GRID_SHOW_PAGE_ARROWS ? minArrowsHeight : 0; - - const indicatorHeight = !this._grid._isFolder - ? minArrowsHeight + ((height - minArrowsHeight) * (1 - opt.APP_GRID_PAGE_HEIGHT_SCALE)) / 2 - : minArrowsHeight + 6; - - return Math.round(indicatorHeight); - }, - - _syncPageIndicators() { - if (!this._container) - return; - - const { value } = this._pageIndicatorsAdjustment; - - const { top, bottom } = this._grid.indicatorsPadding; - const topIndicatorOffset = -top * (1 - value); - const bottomIndicatorOffset = bottom * (1 - value); - - this._previousPageIndicator.translationY = - topIndicatorOffset; - this._nextPageIndicator.translationY = - bottomIndicatorOffset; - - const leftArrowOffset = -top * value; - const rightArrowOffset = bottom * value; - - this._previousPageArrow.translationY = - leftArrowOffset; - this._nextPageArrow.translationY = - rightArrowOffset; - - // Page icons - this._translatePreviousPageIcons(value); - this._translateNextPageIcons(value); - - if (this._grid.nPages > 0) { - this._grid.getItemsAtPage(this._currentPage).forEach(icon => { - icon.translationY = 0; - }); - } - }, - - _translatePreviousPageIcons(value) { - if (this._currentPage === 0) - return; - - const pageHeight = this._grid.layoutManager._pageHeight; - const previousPage = this._currentPage - 1; - const icons = this._grid.getItemsAtPage(previousPage).filter(i => i.visible); - if (icons.length === 0) - return; - - const { top } = this._grid.indicatorsPadding; - const { rowSpacing } = this._grid.layoutManager; - const endIcon = icons[icons.length - 1]; - let iconOffset; - - const currentPageOffset = pageHeight * this._currentPage; - iconOffset = currentPageOffset - endIcon.allocation.y1 - endIcon.width + top - rowSpacing; - - for (const icon of icons) - icon.translationY = iconOffset * value; - }, - - _translateNextPageIcons(value) { - if (this._currentPage >= this._grid.nPages - 1) - return; - - const nextPage = this._currentPage + 1; - const icons = this._grid.getItemsAtPage(nextPage).filter(i => i.visible); - if (icons.length === 0) - return; - - const { bottom } = this._grid.indicatorsPadding; - const { rowSpacing } = this._grid.layoutManager; - let iconOffset; - - const pageOffset = this._pageHeight * nextPage; - iconOffset = pageOffset - icons[0].allocation.y1 - bottom + rowSpacing; - - for (const icon of icons) - icon.translationY = iconOffset * value; - }, - - vfunc_allocate(container, box) { - const indicatorsHeight = this._getIndicatorsHeight(box); - - const pageIndicatorsWidth = 20; // _appDisplay._pageIndicators.width is not stable, 20 is determined by the style - const availWidth = box.get_width() - pageIndicatorsWidth; - const hPadding = Math.round((availWidth - availWidth * opt.APP_GRID_PAGE_WIDTH_SCALE) / 2); - - this._grid.indicatorsPadding = new Clutter.Margin({ - top: indicatorsHeight, - bottom: indicatorsHeight, - left: hPadding + pageIndicatorsWidth, - right: hPadding, - }); - - this._scrollView.allocate(box); - - const topBox = box.copy(); - topBox.y2 = topBox.y1 + indicatorsHeight; - - const bottomBox = box.copy(); - bottomBox.y1 = bottomBox.y2 - indicatorsHeight; - - this._previousPageIndicator.allocate(topBox); - this._previousPageArrow.allocate_align_fill(topBox, - 0.5, 0.5, false, false); - this._nextPageIndicator.allocate(bottomBox); - this._nextPageArrow.allocate_align_fill(bottomBox, - 0.5, 0.5, false, false); - - this._pageHeight = box.get_height(); - - // Center page arrow buttons - this._previousPageArrow.translationX = pageIndicatorsWidth / 2; - this._nextPageArrow.translationX = pageIndicatorsWidth / 2; - // Reset page indicators vertical position - this._nextPageIndicator.translationX = 0; - this._previousPageIndicator.translationX = 0; - }, -}; - -const AppGridCommon = { - _updatePadding() { - const { rowSpacing, columnSpacing } = this.layoutManager; - - const padding = this._indicatorsPadding.copy(); - - padding.left += rowSpacing; - padding.right += rowSpacing; - padding.top += columnSpacing; - padding.bottom += columnSpacing; - - this.layoutManager.pagePadding = padding; - }, -}; - -const FolderIcon = { - after__init() { - this.button_mask = St.ButtonMask.ONE | St.ButtonMask.TWO; - if (shellVersion46) - this.add_style_class_name('app-folder-46'); - else - this.add_style_class_name('app-folder-45'); - }, - - open() { - // Prevent switching page when an item on another page is selected - GLib.idle_add(GLib.PRIORITY_DEFAULT, () => { - // Select folder icon to prevent switching page to the one with currently selected icon - this._parentView._selectAppInternal(this._id); - // Remove key focus from the selected icon to prevent switching page after dropping the removed folder icon on another page of the main grid - this._parentView.grab_key_focus(); - this._ensureFolderDialog(); - this._dialog.popup(); - }); - }, - - vfunc_clicked() { - this.open(); - }, - - _canAccept(source) { - if (!(source instanceof AppDisplay.AppIcon)) - return false; - - const view = _getViewFromIcon(source); - if (!view /* || !(view instanceof AppDisplay.AppDisplay)*/) - return false; - - // Disable this test to allow the user to cancel the current DND by dropping the icon on its original source - /* if (this._folder.get_strv('apps').includes(source.id)) - return false;*/ - - return true; - }, - - acceptDrop(source) { - if (source._sourceItem) - source = source._sourceItem; - - const accepted = AppViewItemCommon.acceptDrop.bind(this)(source); - - if (!accepted) - return false; - - // If the icon is already in the folder (user dropped it back on the same folder), skip re-adding it - if (this._folder.get_strv('apps').includes(source.id)) - return true; - - this._onDragEnd(); - - this.view.addApp(source.app); - - return true; - }, -}; - -const FolderView = { - _createGrid() { - let grid = new FolderGrid(); - grid._view = this; - return grid; - }, - - createFolderIcon(size) { - const layout = new Clutter.GridLayout({ - row_homogeneous: true, - column_homogeneous: true, - }); - - let icon = new St.Widget({ - layout_manager: layout, - x_align: Clutter.ActorAlign.CENTER, - style: `width: ${size}px; height: ${size}px;`, - }); - - const numItems = this._orderedItems.length; - // decide what number of icons switch to 3x3 grid - // APP_GRID_FOLDER_ICON_GRID: 3 -> more than 4 - // : 4 -> more than 8 - const threshold = opt.APP_GRID_FOLDER_ICON_GRID % 3 ? 8 : 4; - const gridSize = opt.APP_GRID_FOLDER_ICON_GRID > 2 && numItems > threshold ? 3 : 2; - const FOLDER_SUBICON_FRACTION = gridSize === 2 ? 0.4 : 0.27; - - let subSize = Math.floor(FOLDER_SUBICON_FRACTION * size); - let rtl = icon.get_text_direction() === Clutter.TextDirection.RTL; - for (let i = 0; i < gridSize * gridSize; i++) { - const style = `width: ${subSize}px; height: ${subSize}px;`; - let bin = new St.Bin({ style, reactive: true }); - bin.pivot_point = new Graphene.Point({ x: 0.5, y: 0.5 }); - if (i < numItems) { - if (!opt.APP_GRID_ACTIVE_PREVIEW) { - bin.child = this._orderedItems[i].app.create_icon_texture(subSize); - } else { - const app = this._orderedItems[i].app; - const child = new AppDisplay.AppIcon(app, { - setSizeManually: true, - showLabel: false, - }); - - child._sourceItem = this._orderedItems[i]; - child._sourceFolder = this; - child.icon.style_class = ''; - child.set_style_class_name(''); - child.icon.set_style('margin: 0; padding: 0;'); - child._dot.set_style('margin-bottom: 1px;'); - child.icon.setIconSize(subSize); - child._canAccept = () => false; - - bin.child = child; - - bin.connect('enter-event', () => { - bin.ease({ - duration: 100, - translation_y: -3, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - }); - bin.connect('leave-event', () => { - bin.ease({ - duration: 100, - translation_y: 0, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - }); - } - } - - layout.attach(bin, rtl ? (i + 1) % gridSize : i % gridSize, Math.floor(i / gridSize), 1, 1); - } - - return icon; - }, - - _loadApps() { - this._apps = []; - const excludedApps = this._folder.get_strv('excluded-apps'); - const appSys = Shell.AppSystem.get_default(); - const addAppId = appId => { - if (excludedApps.includes(appId)) - return; - - if (opt.APP_GRID_EXCLUDE_FAVORITES && this._appFavorites.isFavorite(appId)) - return; - - const app = appSys.lookup_app(appId); - if (!app) - return; - - if (opt.APP_GRID_EXCLUDE_RUNNING) { - const runningApps = Shell.AppSystem.get_default().get_running().map(a => a.id); - if (runningApps.includes(appId)) - return; - } - - if (!this._parentalControlsManager.shouldShowApp(app.get_app_info())) - return; - - if (this._apps.indexOf(app) !== -1) - return; - - this._apps.push(app); - }; - - const folderApps = this._folder.get_strv('apps'); - folderApps.forEach(addAppId); - - const folderCategories = this._folder.get_strv('categories'); - const appInfos = this._parentView.getAppInfos(); - appInfos.forEach(appInfo => { - let appCategories = /* AppDisplay.*/_getCategories(appInfo); - if (!_listsIntersect(folderCategories, appCategories)) - return; - - addAppId(appInfo.get_id()); - }); - - let items = []; - this._apps.forEach(app => { - let icon = this._items.get(app.get_id()); - if (!icon) - icon = new AppDisplay.AppIcon(app); - - items.push(icon); - }); - - return items; - }, - - acceptDrop(source) { - /* if (!BaseAppViewCommon.acceptDrop.bind(this)(source)) - return false;*/ - if (opt.APP_FOLDER_ORDER) - return false; - if (source._sourceItem) - source = source._sourceItem; - - if (!this._acceptDropCommon(source)) - return false; - - const folderApps = this._orderedItems.map(item => item.id); - this._folder.set_strv('apps', folderApps); - - return true; - }, -}; - -const FolderGrid = GObject.registerClass({ - // Registered name should be unique - GTypeName: `FolderGrid${Math.floor(Math.random() * 1000)}`, -}, class FolderGrid extends AppDisplay.AppGrid { - _init() { - super._init({ - allow_incomplete_pages: false, - // For adaptive size (0), set the numbers high enough to fit all the icons - // to avoid splitting the icons to pages upon creating the grid - columns_per_page: 20, - rows_per_page: 20, - page_halign: Clutter.ActorAlign.CENTER, - page_valign: Clutter.ActorAlign.CENTER, - }); - this.layoutManager._isFolder = true; - this._isFolder = true; - const spacing = opt.APP_GRID_FOLDER_SPACING; - this.set_style(`column-spacing: ${spacing}px; row-spacing: ${spacing}px;`); - this.layoutManager.fixedIconSize = opt.APP_GRID_FOLDER_ICON_SIZE; - - this.setGridModes([ - { - columns: 20, - rows: 20, - }, - ]); - } - - _updatePadding() { - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const padding = this._indicatorsPadding.copy(); - const pageIndicatorSize = opt.ORIENTATION - ? this._view._pageIndicators.get_preferred_width(1000)[1] / scaleFactor - : this._view._pageIndicators.get_preferred_height(1000)[1] / scaleFactor; - Math.round(Math.min(...this._view._pageIndicators.get_size()));// / scaleFactor);// ~28; - padding.left = opt.ORIENTATION ? pageIndicatorSize : 0; - padding.right = 0; - padding.top = opt.ORIENTATION ? 0 : pageIndicatorSize; - padding.bottom = 0; - this.layoutManager.pagePadding = padding; - } -}); - - -const FOLDER_DIALOG_ANIMATION_TIME = 200; // AppDisplay.FOLDER_DIALOG_ANIMATION_TIME -const AppFolderDialog = { - // injection to _init() - after__init() { - // GS 46 changed the aligning to CENTER which restricts max folder dialog size - this._viewBox.set({ - x_align: Clutter.ActorAlign.FILL, - y_align: Clutter.ActorAlign.FILL, - }); - - // delegate this dialog to the FolderIcon._view - // so its _createFolderIcon function can update the dialog if folder content changed - this._view._dialog = this; - - // right click into the folder popup should close it - this.child.reactive = true; - const clickAction = new Clutter.ClickAction(); - clickAction.connect('clicked', act => { - if (act.get_button() === Clutter.BUTTON_PRIMARY) - return Clutter.EVENT_STOP; - const [x, y] = clickAction.get_coords(); - const actor = global.stage.get_actor_at_pos(Clutter.PickMode.ALL, x, y); - // if it's not entry for editing folder title - if (actor !== this._entry) - this.popdown(); - return Clutter.EVENT_STOP; - }); - - this.child.add_action(clickAction); - }, - - after__addFolderNameEntry() { - // edit-folder-button class has been replaced with icon-button class which is not transparent in 46 - this._editButton.add_style_class_name('edit-folder-button'); - - // Edit button - this._removeButton = new St.Button({ - style_class: 'icon-button edit-folder-button', - button_mask: St.ButtonMask.ONE, - toggle_mode: false, - reactive: true, - can_focus: true, - x_align: Clutter.ActorAlign.END, - y_align: Clutter.ActorAlign.CENTER, - child: new St.Icon({ - icon_name: 'user-trash-symbolic', - icon_size: 16, - }), - }); - - this._removeButton.connect('clicked', () => { - if (Date.now() - this._removeButton._lastClick < Clutter.Settings.get_default().double_click_time) { - // Close dialog to avoid crashes - this._isOpen = false; - this._grabHelper.ungrab({ actor: this }); - this.emit('open-state-changed', false); - this.hide(); - this._popdownCallbacks.forEach(func => func()); - this._popdownCallbacks = []; - _appDisplay.ease({ - opacity: 255, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - - // Reset all keys to delete the relocatable schema - this._view._deletingFolder = true; // Upstream property - let keys = this._folder.settings_schema.list_keys(); - for (const key of keys) - this._folder.reset(key); - - let settings = new Gio.Settings({ schema_id: 'org.gnome.desktop.app-folders' }); - let folders = settings.get_strv('folder-children'); - folders.splice(folders.indexOf(this._view._id), 1); - - // remove all abandoned folders (usually my own garbage and unwanted default folders...) - /* const appFolders = _appDisplay._folderIcons.map(icon => icon._id); - folders.forEach(folder => { - if (!appFolders.includes(folder)) { - folders.splice(folders.indexOf(folder._id), 1); - } - });*/ - settings.set_strv('folder-children', folders); - - this._view._deletingFolder = false; - return; - } - this._removeButton._lastClick = Date.now(); - }); - - this._entryBox.add_child(this._removeButton); - this._entryBox.set_child_at_index(this._removeButton, 0); - - this._closeButton = new St.Button({ - style_class: 'icon-button edit-folder-button', - button_mask: St.ButtonMask.ONE, - toggle_mode: false, - reactive: true, - can_focus: true, - x_align: Clutter.ActorAlign.END, - y_align: Clutter.ActorAlign.CENTER, - child: new St.Icon({ - icon_name: 'window-close-symbolic', - icon_size: 16, - }), - }); - - this._closeButton.connect('clicked', () => { - this.popdown(); - }); - - this._entryBox.add_child(this._closeButton); - }, - - popup() { - if (this._isOpen) - return; - - this._isOpen = this._grabHelper.grab({ - actor: this, - focus: this._editButton, - onUngrab: () => this.popdown(), - }); - - if (!this._isOpen) - return; - - this.get_parent().set_child_above_sibling(this, null); - - // _zoomAndFadeIn() is called from the dialog's allocate() - this._needsZoomAndFade = true; - - this.show(); - // force update folder size - this._folderAreaBox = null; - this._updateFolderSize(); - - this.emit('open-state-changed', true); - }, - - _setupPopdownTimeout() { - if (this._popdownTimeoutId > 0) - return; - - // This timeout is handled in the original code and removed in _onDestroy() - // All dialogs are destroyed on extension disable() - this._popdownTimeoutId = - GLib.timeout_add(GLib.PRIORITY_DEFAULT, 500, () => { - this._popdownTimeoutId = 0; - // Following line fixes upstream bug - // https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/6164 - this._view._onDragEnd(); - this.popdown(); - return GLib.SOURCE_REMOVE; - }); - }, - - vfunc_allocate(box) { - this._updateFolderSize(); - - // super.allocate(box) - St.Bin.prototype.vfunc_allocate.bind(this)(box); - - // Override any attempt to resize the folder dialog, that happens when some child gets wild - // Re-allocate the child only if necessary, because it terminates grid animations - if (this._width && this._height && (this._width !== this.child.width || this._height !== this.child.height)) - this._allocateChild(); - - // We can only start zooming after receiving an allocation - if (this._needsZoomAndFade) - this._zoomAndFadeIn(); - }, - - _allocateChild() { - const childBox = new Clutter.ActorBox(); - childBox.set_size(this._width, this._height); - this.child.allocate(childBox); - }, - - // Note that the appDisplay may be off-screen so its coordinates may be shifted - // However, for _updateFolderSize() it doesn't matter - // and when _zoomAndFadeIn() is called, appDisplay is on the right place - _getFolderAreaBox() { - const appDisplay = this._source._parentView; - const folderAreaBox = appDisplay.get_allocation_box().copy(); - const searchEntryHeight = opt.SHOW_SEARCH_ENTRY ? Main.overview._overview.controls._searchEntryBin.height : 0; - folderAreaBox.y1 -= searchEntryHeight; - - // _zoomAndFadeIn() needs an absolute position within a multi-monitor workspace - const monitorGeometry = global.display.get_monitor_geometry(global.display.get_primary_monitor()); - folderAreaBox.x1 += monitorGeometry.x; - folderAreaBox.x2 += monitorGeometry.x; - folderAreaBox.y1 += monitorGeometry.y; - folderAreaBox.y2 += monitorGeometry.y; - - return folderAreaBox; - }, - - _updateFolderSize() { - const view = this._view; - const nItems = view._orderedItems.length; - const [firstItem] = view._grid.layoutManager._container; - if (!firstItem) - return; - - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const margin = 18; // see stylesheet .app-folder-dialog-container; - - const folderAreaBox = this._getFolderAreaBox(); - - const maxDialogWidth = folderAreaBox.get_width() / scaleFactor; - const maxDialogHeight = folderAreaBox.get_height() / scaleFactor; - - // We can't build folder if the available space is not available - if (!isFinite(maxDialogWidth) || !isFinite(maxDialogHeight) || !maxDialogWidth || !maxDialogHeight) - return; - - // We don't need to recalculate grid if nothing changed - if ( - this._folderAreaBox?.get_width() === folderAreaBox.get_width() && - this._folderAreaBox?.get_height() === folderAreaBox.get_height() && - nItems === this._nItems - ) - return; - - const layoutManager = view._grid.layoutManager; - const spacing = opt.APP_GRID_FOLDER_SPACING; - const padding = 40; - - const titleBoxHeight = - Math.round(this._entryBox.get_preferred_height(-1)[1] / scaleFactor); // ~75 - const minDialogWidth = Math.max(640, - Math.round(this._entryBox.get_preferred_width(-1)[1] / scaleFactor + 2 * margin)); - const navigationArrowsSize = // padding + one arrow width is sufficient for both arrows - Math.round(view._nextPageArrow.get_preferred_width(-1)[1] / scaleFactor); - const pageIndicatorSize = - Math.round(Math.min(...view._pageIndicators.get_size()) / scaleFactor);// ~28; - const horizontalNavigation = opt.ORIENTATION ? pageIndicatorSize : navigationArrowsSize; // either add padding or arrows - const verticalNavigation = opt.ORIENTATION ? navigationArrowsSize : pageIndicatorSize; - - // Horizontal size - const baseWidth = horizontalNavigation + 3 * padding + 2 * margin; - const maxGridPageWidth = maxDialogWidth - baseWidth; - // Vertical size - const baseHeight = titleBoxHeight + verticalNavigation + 2 * padding + 2 * margin; - const maxGridPageHeight = maxDialogHeight - baseHeight; - - // Will be updated to the actual value later - let itemPadding = 55; - const minItemSize = 48 + itemPadding; - - let columns = opt.APP_GRID_FOLDER_COLUMNS; - let rows = opt.APP_GRID_FOLDER_ROWS; - const maxColumns = columns ? columns : 100; - const maxRows = rows ? rows : 100; - - // Find best icon size - let iconSize = opt.APP_GRID_FOLDER_ICON_SIZE < 0 ? opt.APP_GRID_FOLDER_ICON_SIZE_DEFAULT : opt.APP_GRID_FOLDER_ICON_SIZE; - if (opt.APP_GRID_FOLDER_ICON_SIZE === -1) { - let maxIconSize; - if (columns) { - const maxItemWidth = (maxGridPageWidth - (columns - 1) * opt.APP_GRID_FOLDER_SPACING) / columns; - maxIconSize = maxItemWidth - itemPadding; - } - if (rows) { - const maxItemHeight = (maxGridPageHeight - (rows - 1) * spacing) / rows; - maxIconSize = Math.min(maxItemHeight - itemPadding, maxIconSize); - } - - if (maxIconSize) { - // We only need sizes from the default to the smallest - let iconSizes = Object.values(IconSize).sort((a, b) => b - a); - iconSizes = iconSizes.slice(iconSizes.indexOf(iconSize)); - for (const size of iconSizes) { - iconSize = size; - if (iconSize <= maxIconSize) - break; - } - } - } - - if ((!columns && !rows) || opt.APP_GRID_FOLDER_ICON_SIZE !== -1) { - columns = Math.ceil(Math.sqrt(nItems)); - rows = columns; - if (columns * (columns - 1) >= nItems) { - rows = columns - 1; - } else if ((columns + 1) * (columns - 1) >= nItems) { - rows = columns - 1; - columns += 1; - } - } else if (columns && !rows) { - rows = Math.ceil(nItems / columns); - } else if (rows && !columns) { - columns = Math.ceil(nItems / rows); - } - - columns = Math.clamp(columns, 1, maxColumns); - columns = Math.min(nItems, columns); - rows = Math.clamp(rows, 1, maxRows); - - let itemSize = iconSize + itemPadding; - // First run sets the grid before we can read the real icon size - // so we estimate the size from default properties - // and correct it in the second run - if (this.realized) { - firstItem.icon.setIconSize(iconSize); - // Item height is inconsistent because it depends on its label height - const [, firstItemWidth] = firstItem.get_preferred_width(-1); - const realSize = firstItemWidth / scaleFactor; - itemSize = realSize; - itemPadding = realSize - iconSize; - } - - const gridWidth = columns * (itemSize + spacing); - let width = gridWidth + baseWidth; - const gridHeight = rows * (itemSize + spacing); - let height = gridHeight + baseHeight; - - // Folder must fit the appDisplay area plus searchEntryBin if visible - // reduce columns/rows if needed - while (height > maxDialogHeight && rows > 1) { - height -= itemSize + spacing; - rows -= 1; - } - - while (width > maxDialogWidth && columns > 1) { - width -= itemSize + spacing; - columns -= 1; - } - - // Try to compensate for the previous reduction if there is a space - while ((nItems > columns * rows) && ((width + (itemSize + spacing)) <= maxDialogWidth) && (columns < maxColumns)) { - width += itemSize + spacing; - columns += 1; - } - - // remove columns that cannot be displayed - if (((columns * minItemSize + (columns - 1) * spacing)) > maxDialogWidth) - columns = Math.floor(maxDialogWidth / (minItemSize + spacing)); - - while ((nItems > columns * rows) && ((height + (itemSize + spacing)) <= maxDialogHeight) && (rows < maxRows)) { - height += itemSize + spacing; - rows += 1; - } - // remove rows that cannot be displayed - if ((((rows * minItemSize + (rows - 1) * spacing))) > maxDialogHeight) - rows = Math.floor(maxDialogWidth / (minItemSize + spacing)); - - // remove size for rows that are empty - const rowsNeeded = Math.ceil(nItems / columns); - if (rows > rowsNeeded) { - height -= (rows - rowsNeeded) * (itemSize + spacing); - rows -= rows - rowsNeeded; - } - - // Remove space reserved for page controls and indicator if not used - if (rows * columns >= nItems) { - width -= horizontalNavigation; - height -= verticalNavigation; - } - - width = Math.clamp(width, minDialogWidth, maxDialogWidth); - height = Math.min(height, maxDialogHeight); - - layoutManager.columns_per_page = columns; - layoutManager.rows_per_page = rows; - - layoutManager.fixedIconSize = iconSize; - - - // Store data for further use - this._width = width * scaleFactor; - this._height = height * scaleFactor; - this._folderAreaBox = folderAreaBox; - this._nItems = nItems; - - // Set fixed dialog size to prevent size instability - this.child.set_size(this._width, this._height); - this._viewBox.set_style(`width: ${this._width - 2 * margin}px; height: ${this._height - 2 * margin}px;`); - this._viewBox.set_size(this._width - 2 * margin, this._height - 2 * margin); - - view._redisplay(); - }, - - _zoomAndFadeIn() { - let [sourceX, sourceY] = - this._source.get_transformed_position(); - let [dialogX, dialogY] = - this.child.get_transformed_position(); - - const sourceCenterX = sourceX + this._source.width / 2; - const sourceCenterY = sourceY + this._source.height / 2; - - // this. covers the whole screen - let dialogTargetX = dialogX; - let dialogTargetY = dialogY; - - const appDisplay = this._source._parentView; - - const folderAreaBox = this._getFolderAreaBox(); - - let folderAreaX = folderAreaBox.x1; - let folderAreaY = folderAreaBox.y1; - const folderAreaWidth = folderAreaBox.get_width(); - const folderAreaHeight = folderAreaBox.get_height(); - const folder = this.child; - - if (opt.APP_GRID_FOLDER_CENTER) { - dialogTargetX = folderAreaX + folderAreaWidth / 2 - folder.width / 2; - dialogTargetY = folderAreaY + (folderAreaHeight / 2 - folder.height / 2) / 2; - } else { - const { pagePadding } = appDisplay._grid.layoutManager; - const hPadding = (pagePadding.left + pagePadding.right) / 2; - const vPadding = (pagePadding.top + pagePadding.bottom) / 2; - const minX = Math.min(folderAreaX + hPadding, folderAreaX + (folderAreaWidth - folder.width) / 2); - const maxX = Math.max(folderAreaX + folderAreaWidth - hPadding - folder.width, folderAreaX + folderAreaWidth / 2 - folder.width / 2); - const minY = Math.min(folderAreaY + vPadding, folderAreaY + (folderAreaHeight - folder.height) / 2); - const maxY = Math.max(folderAreaY + folderAreaHeight - vPadding - folder.height, folderAreaY + folderAreaHeight / 2 - folder.height / 2); - - dialogTargetX = sourceCenterX - folder.width / 2; - dialogTargetX = Math.clamp(dialogTargetX, minX, maxX); - dialogTargetY = sourceCenterY - folder.height / 2; - dialogTargetY = Math.clamp(dialogTargetY, minY, maxY); - - // keep the dialog in the appDisplay area - dialogTargetX = Math.clamp( - dialogTargetX, - folderAreaX, - folderAreaX + folderAreaWidth - folder.width - ); - - dialogTargetY = Math.clamp( - dialogTargetY, - folderAreaY, - folderAreaY + folderAreaHeight - folder.height - ); - } - - const dialogOffsetX = Math.round(dialogTargetX - dialogX); - const dialogOffsetY = Math.round(dialogTargetY - dialogY); - - this.child.set({ - translation_x: sourceX - dialogX, - translation_y: sourceY - dialogY, - scale_x: this._source.width / this.child.width, - scale_y: this._source.height / this.child.height, - opacity: 0, - }); - - this.child.ease({ - translation_x: dialogOffsetX, - translation_y: dialogOffsetY, - scale_x: 1, - scale_y: 1, - opacity: 255, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - - appDisplay.ease({ - opacity: 0, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - - if (opt.SHOW_SEARCH_ENTRY) { - Main.overview.searchEntry.ease({ - opacity: 0, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - - this._needsZoomAndFade = false; - - if (this._sourceMappedId === 0) { - this._sourceMappedId = this._source.connect( - 'notify::mapped', this._zoomAndFadeOut.bind(this)); - } - }, - - _zoomAndFadeOut() { - if (!this._isOpen) - return; - - if (!this._source.mapped) { - this.hide(); - return; - } - - // if the dialog was shown silently, skip animation - if (this.scale_y < 1) { - this._needsZoomAndFade = false; - this.hide(); - this._popdownCallbacks.forEach(func => func()); - this._popdownCallbacks = []; - return; - } - - let [sourceX, sourceY] = - this._source.get_transformed_position(); - let [dialogX, dialogY] = - this.child.get_transformed_position(); - - this.child.ease({ - translation_x: sourceX - dialogX + this.child.translation_x, - translation_y: sourceY - dialogY + this.child.translation_y, - scale_x: this._source.width / this.child.width, - scale_y: this._source.height / this.child.height, - opacity: 0, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_IN_QUAD, - onComplete: () => { - this.child.set({ - translation_x: 0, - translation_y: 0, - scale_x: 1, - scale_y: 1, - opacity: 255, - }); - this.hide(); - - this._popdownCallbacks.forEach(func => func()); - this._popdownCallbacks = []; - }, - }); - - const appDisplay = this._source._parentView; - appDisplay.ease({ - opacity: 255, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_IN_QUAD, - }); - - if (opt.SHOW_SEARCH_ENTRY) { - Main.overview.searchEntry.ease({ - opacity: 255, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_IN_QUAD, - }); - } - - this._needsZoomAndFade = false; - }, - - _setLighterBackground(lighter) { - let opacity = 255; - if (this._isOpen) - opacity = lighter ? 20 : 0; - - _appDisplay.ease({ - opacity, - duration: FOLDER_DIALOG_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - }, - - vfunc_key_press_event(event) { - if (global.focus_manager.navigate_from_event(event)) - return Clutter.EVENT_STOP; - return Clutter.EVENT_PROPAGATE; - }, - - _showFolderLabel() { - if (this._editButton.checked) - this._editButton.checked = false; - - this._maybeUpdateFolderName(); - this._switchActor(this._entry, this._folderNameLabel); - // This line has been added in 47 to fix focus after editing the folder name - this.navigate_focus(this, St.DirectionType.TAB_FORWARD, false); - }, -}; - -const AppIcon = { - after__init() { - // update the app label behavior - this._updateMultiline(); - }, - - // avoid accepting by placeholder when dragging active preview - // and also by icon if usage sorting is used - _canAccept(source) { - if (source._sourceItem) - source = source._sourceItem; - - // Folders in folder are not supported - if (!(_getViewFromIcon(this) instanceof AppDisplay.AppDisplay) || !this.opacity) - return false; - - const view = /* AppDisplay.*/_getViewFromIcon(source); - return source !== this && - (source instanceof this.constructor) && - // Include drops from folders - // (view instanceof AppDisplay.AppDisplay && - (view && - !opt.APP_GRID_USAGE); - }, -}; - -const AppViewItemCommon = { - _updateMultiline() { - const { label } = this.icon; - if (label) - label.opacity = 255; - if (!this._expandTitleOnHover || !this.icon.label) - return; - - const { clutterText } = label; - - const isHighlighted = this.has_key_focus() || this.hover || this._forcedHighlight; - - if (opt.APP_GRID_NAMES_MODE === 2 && this._expandTitleOnHover) { // !_expandTitleOnHover indicates search result icon - label.opacity = isHighlighted || !this.app ? 255 : 0; - } - if (isHighlighted) - this.get_parent()?.set_child_above_sibling(this, null); - - if (!opt.APP_GRID_NAMES_MODE) { - const layout = clutterText.get_layout(); - if (!layout.is_wrapped() && !layout.is_ellipsized()) - return; - } - - label.remove_transition('allocation'); - - const id = label.connect('notify::allocation', () => { - label.restore_easing_state(); - label.disconnect(id); - }); - - const expand = opt.APP_GRID_NAMES_MODE === 1 || this._forcedHighlight || this.hover || this.has_key_focus(); - - label.save_easing_state(); - label.set_easing_duration(expand - ? APP_ICON_TITLE_EXPAND_TIME - : APP_ICON_TITLE_COLLAPSE_TIME); - clutterText.set({ - line_wrap: expand, - line_wrap_mode: expand ? Pango.WrapMode.WORD_CHAR : Pango.WrapMode.NONE, - ellipsize: expand ? Pango.EllipsizeMode.NONE : Pango.EllipsizeMode.END, - }); - }, - - // support active preview icons - acceptDrop(source, _actor, x) { - if (opt.APP_GRID_USAGE) - return DND.DragMotionResult.NO_DROP; - - this._setHoveringByDnd(false); - - if (!this._canAccept(source)) - return false; - - if (this._withinLeeways(x)) - return false; - - // added - remove app from the source folder after dnd to other folder - let view = /* AppDisplay.*/_getViewFromIcon(source); - if (view instanceof AppDisplay.FolderView) - view.removeApp(source.app); - - return true; - }, - -}; - -const PageIndicatorsCommon = { - after_setNPages() { - this.visible = true; - this.opacity = this._nPages > 1 ? 255 : 0; - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/appFavorites.js b/extensions/46/vertical-workspaces/lib/appFavorites.js deleted file mode 100644 index 977e65a..0000000 --- a/extensions/46/vertical-workspaces/lib/appFavorites.js +++ /dev/null @@ -1,79 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * appFavorites.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import * as AppFavorites from 'resource:///org/gnome/shell/ui/appFavorites.js'; - -let Me; -let opt; - -export const AppFavoritesModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('appFavoritesModule'); - - // if notifications are enabled no override is needed - reset = reset || !this.moduleEnabled || opt.SHOW_FAV_NOTIFICATION; - - // don't touch original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) { - this.moduleEnabled = false; - console.debug(' AppFavoritesModule - Keeping untouched'); - } - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - // use actual instance instead of prototype - this._overrides.addOverride('AppFavorites', AppFavorites.getAppFavorites(), AppFavoritesCommon); - - console.debug(' AppFavoritesModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - console.debug(' AppFavoritesModule - Deactivated'); - } -}; - -const AppFavoritesCommon = { - addFavoriteAtPos(appId, pos) { - this._addFavorite(appId, pos); - }, - - removeFavorite(appId) { - this._removeFavorite(appId); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/dash.js b/extensions/46/vertical-workspaces/lib/dash.js deleted file mode 100644 index c7ebbff..0000000 --- a/extensions/46/vertical-workspaces/lib/dash.js +++ /dev/null @@ -1,1307 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * dash.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022-2024 - * @license GPL-3.0 - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import GLib from 'gi://GLib'; -import Meta from 'gi://Meta'; -import Shell from 'gi://Shell'; -import St from 'gi://St'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Dash from 'resource:///org/gnome/shell/ui/dash.js'; -import * as AppDisplay from 'resource:///org/gnome/shell/ui/appDisplay.js'; -import * as AppFavorites from 'resource:///org/gnome/shell/ui/appFavorites.js'; -import * as AppMenu from 'resource:///org/gnome/shell/ui/appMenu.js'; -import * as BoxPointer from 'resource:///org/gnome/shell/ui/boxpointer.js'; -import * as DND from 'resource:///org/gnome/shell/ui/dnd.js'; -import * as PopupMenu from 'resource:///org/gnome/shell/ui/popupMenu.js'; - -let Me; -let opt; -// gettext -let _; - -let _moduleEnabled; -let _timeouts; - -// added values to achieve a better ability to scale down according to available space -export const BaseIconSizes = [16, 24, 32, 40, 44, 48, 56, 64, 72, 80, 96, 112, 128]; - -const DASH_ITEM_LABEL_SHOW_TIME = 150; - -const shellVersion46 = !Clutter.Container; // Container has been removed in 46 - -export const DashModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - _ = Me.gettext; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - this._horizontalWorkId = null; - this._verticalWorkId = null; - this._showAppsIconBtnPressId = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - _ = null; - } - - update(reset) { - this._removeTimeouts(); - - this.moduleEnabled = opt.get('dashModule'); - const conflict = !!(Me.Util.getEnabledExtensions('dash-to-dock').length || - Me.Util.getEnabledExtensions('dash2dock').length || - Me.Util.getEnabledExtensions('ubuntu-dock').length || - Me.Util.getEnabledExtensions('dash-to-panel').length); - - if (conflict && !reset) - console.warn(`[${Me.metadata.name}] Warning: "Dash" module disabled due to potential conflict with another extension`); - - reset = reset || !this.moduleEnabled || conflict; - this._conflict = conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' DashModule - Keeping untouched'); - } - - updateStyle(dash) { - if (opt.DASH_BG_LIGHT) - dash._background.add_style_class_name('dash-background-light'); - else - dash._background.remove_style_class_name('dash-background-light'); - - dash._background.opacity = opt.DASH_BG_OPACITY; - let radius = opt.DASH_BG_RADIUS; - if (radius) { - let style; - switch (opt.DASH_POSITION) { - case 1: - style = opt.DASH_BG_GS3_STYLE ? `border-radius: ${radius}px 0 0 ${radius}px;` : `border-radius: ${radius}px;`; - break; - case 3: - style = opt.DASH_BG_GS3_STYLE ? `border-radius: 0 ${radius}px ${radius}px 0;` : `border-radius: ${radius}px;`; - break; - default: - style = `border-radius: ${radius}px;`; - } - dash._background.set_style(style); - } else { - dash._background.set_style(''); - } - } - - _activateModule() { - _moduleEnabled = true; - _timeouts = {}; - const dash = Main.overview._overview._controls.layoutManager._dash; - - if (!this._originalWorkId) - this._originalWorkId = dash._workId; - - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._resetStyle(dash); - this.updateStyle(dash); - - this._overrides.addOverride('DashItemContainer', Dash.DashItemContainer.prototype, DashItemContainerCommon); - this._overrides.addOverride('DashCommon', Dash.Dash.prototype, DashCommon); - this._overrides.addOverride('AppIcon', AppDisplay.AppIcon.prototype, AppIconCommon); - this._overrides.addOverride('DashIcon', Dash.DashIcon.prototype, DashIconCommon); - this._overrides.addOverride('AppMenu', AppMenu.AppMenu.prototype, AppMenuCommon); - - if (shellVersion46) - dash.add_style_class_name('dash-46'); - - if (opt.DASH_VERTICAL) { - // this._overrides.addOverride('Dash', Dash.Dash.prototype, DashVerticalOverride); - dash.add_style_class_name(shellVersion46 - ? 'vertical-46' - : 'vertical' - ); - - this._setOrientation(Clutter.Orientation.VERTICAL); - } else { - this._setOrientation(Clutter.Orientation.HORIZONTAL); - } - - if (opt.DASH_VERTICAL && opt.DASH_BG_GS3_STYLE) { - if (opt.DASH_LEFT) { - dash.add_style_class_name(shellVersion46 - ? 'vertical-46-gs3-left' - : 'vertical-gs3-left'); - } else if (opt.DASH_RIGHT) { - dash.add_style_class_name(shellVersion46 - ? 'vertical-46-gs3-right' - : 'vertical-gs3-right'); - } - } else { - dash.remove_style_class_name('vertical-gs3-left'); - dash.remove_style_class_name('vertical-gs3-right'); - dash.remove_style_class_name('vertical-46-gs3-left'); - dash.remove_style_class_name('vertical-46-gs3-right'); - } - - if (!this._customWorkId) - this._customWorkId = Main.initializeDeferredWork(dash._box, dash._redisplay.bind(dash)); - dash._workId = this._customWorkId; - - this._moveDashAppGridIcon(); - this._connectShowAppsIcon(); - - dash.visible = opt.DASH_VISIBLE; - // dash._background.add_style_class_name('dash-background-reduced'); - dash._queueRedisplay(); - - if (opt.DASH_ISOLATE_WS && !this._wmSwitchWsConId) { - this._wmSwitchWsConId = global.windowManager.connect('switch-workspace', () => dash._queueRedisplay()); - this._newWindowConId = global.display.connect_after('window-created', () => dash._queueRedisplay()); - } - - console.debug(' DashModule - Activated'); - } - - _disableModule() { - const dash = Main.overview._overview._controls.layoutManager._dash; - - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - dash._workId = this._originalWorkId; - - if (this._wmSwitchWsConId) { - global.windowManager.disconnect(this._wmSwitchWsConId); - this._wmSwitchWsConId = 0; - } - if (this._newWindowConId) { - global.windowManager.disconnect(this._newWindowConId); - this._newWindowConId = 0; - } - - const reset = true; - this._setOrientation(Clutter.Orientation.HORIZONTAL); - this._moveDashAppGridIcon(reset); - this._connectShowAppsIcon(reset); - - this._resetStyle(dash); - dash.visible = !this._conflict; - dash._background.opacity = 255; - - _moduleEnabled = false; - console.debug(' DashModule - Disabled'); - } - - _resetStyle(dash) { - dash.remove_style_class_name('dash-46'); - dash.remove_style_class_name('vertical'); - dash.remove_style_class_name('vertical-46'); - dash.remove_style_class_name('vertical-gs3-left'); - dash.remove_style_class_name('vertical-gs3-right'); - dash.remove_style_class_name('vertical-46-gs3-left'); - dash.remove_style_class_name('vertical-46-gs3-right'); - dash.remove_style_class_name('vertical-left'); - dash.remove_style_class_name('vertical-right'); - dash._background.remove_style_class_name('dash-background-light'); - dash._background.remove_style_class_name('dash-background-reduced'); - dash._background.set_style(''); - } - - _removeTimeouts() { - if (_timeouts) { - Object.values(_timeouts).forEach(t => { - if (t) - GLib.source_remove(t); - }); - _timeouts = null; - } - } - - _setOrientation(orientation, dash) { - dash = dash ?? Main.overview._overview._controls.layoutManager._dash; - - dash._box.layout_manager.orientation = orientation; - dash._dashContainer.layout_manager.orientation = orientation; - dash._dashContainer.y_expand = !orientation; - dash._dashContainer.x_expand = !!orientation; - dash.x_align = orientation ? Clutter.ActorAlign.START : Clutter.ActorAlign.CENTER; - dash.y_align = orientation ? Clutter.ActorAlign.CENTER : Clutter.ActorAlign.FILL; - - let sizerBox = dash._background.get_children()[0]; - sizerBox.clear_constraints(); - sizerBox.add_constraint(new Clutter.BindConstraint({ - source: dash._showAppsIcon.icon, - coordinate: orientation ? Clutter.BindCoordinate.WIDTH : Clutter.BindCoordinate.HEIGHT, - })); - sizerBox.add_constraint(new Clutter.BindConstraint({ - source: dash._dashContainer, - coordinate: orientation ? Clutter.BindCoordinate.HEIGHT : Clutter.BindCoordinate.WIDTH, - })); - dash._box.remove_all_children(); - dash._separator = null; - dash._queueRedisplay(); - dash._adjustIconSize(); - } - - _moveDashAppGridIcon(reset = false) { - // move dash app grid icon to the front - const dash = Main.overview._overview._controls.layoutManager._dash; - - const appIconPosition = opt.get('showAppsIconPosition'); - dash._showAppsIcon.remove_style_class_name('show-apps-icon-vertical-hide'); - dash._showAppsIcon.remove_style_class_name('show-apps-icon-horizontal-hide'); - dash._showAppsIcon.opacity = 255; - if (!reset && appIconPosition === 0) // 0 - start - dash._dashContainer.set_child_at_index(dash._showAppsIcon, 0); - if (reset || appIconPosition === 1) { // 1 - end - const index = dash._dashContainer.get_children().length - 1; - dash._dashContainer.set_child_at_index(dash._showAppsIcon, index); - } - if (!reset && appIconPosition === 2) { // 2 - hide - const style = opt.DASH_VERTICAL ? 'show-apps-icon-vertical-hide' : 'show-apps-icon-horizontal-hide'; - dash._showAppsIcon.add_style_class_name(style); - // for some reason even if the icon height in vertical mode should be set to 0 by the style, it stays visible in full size returning height 1px - dash._showAppsIcon.opacity = 0; - } - } - - _connectShowAppsIcon(reset = false, dash) { - dash = dash ?? Main.overview._overview._controls.layoutManager._dash; - if (!reset) { - if (this._showAppsIconBtnPressId || Me.Util.dashIsDashToDock()) { - // button is already connected || dash is Dash to Dock - return; - } - dash._showAppsIcon.reactive = true; - this._showAppsIconBtnPressId = dash._showAppsIcon.connect('button-press-event', (actor, event) => { - const button = event.get_button(); - if (button === Clutter.BUTTON_MIDDLE) - Me.Util.openPreferences(); - else if (button === Clutter.BUTTON_SECONDARY) - Me.Util.activateSearchProvider(Me.WSP_PREFIX); - else - return Clutter.EVENT_PROPAGATE; - return Clutter.EVENT_STOP; - }); - } else if (this._showAppsIconBtnPressId) { - dash._showAppsIcon.disconnect(this._showAppsIconBtnPressId); - this._showAppsIconBtnPressId = 0; - dash._showAppsIcon.reactive = false; - } - } -}; - -function getAppFromSource(source) { - if (source instanceof AppDisplay.AppIcon) - return source.app; - else - return null; -} - -const DashItemContainerCommon = { - // move labels according dash position - showLabel() { - if (!this._labelText) - return; - - const windows = this.child.app?.get_windows(); - const recentWindowTitle = windows && windows.length ? windows[0].get_title() : ''; - const windowCount = this.child.app?.get_windows().length; - let labelSuffix = ''; - if (windowCount > 1) - labelSuffix = ` (${windowCount})`; - if (recentWindowTitle && recentWindowTitle !== this._labelText) - labelSuffix += `\n ${recentWindowTitle}`; - - this.label.set_text(this._labelText + labelSuffix); - - this.label.opacity = 0; - this.label.show(); - - let [stageX, stageY] = this.get_transformed_position(); - - const itemWidth = this.allocation.get_width(); - const itemHeight = this.allocation.get_height(); - - const labelWidth = this.label.get_width(); - const labelHeight = this.label.get_height(); - let xOffset = Math.floor((itemWidth - labelWidth) / 2); - let x = Math.clamp(stageX + xOffset, 0, global.stage.width - labelWidth); - const primaryMonitor = global.display.get_monitor_geometry(global.display.get_primary_monitor()); - x = Math.clamp(x, primaryMonitor.x, primaryMonitor.x + primaryMonitor.width - labelWidth); - - let node = this.label.get_theme_node(); - let y; - - if (opt.DASH_TOP) { - const yOffset = itemHeight + (shellVersion46 ? 0 : -3); - y = stageY + yOffset; - } else if (opt.DASH_BOTTOM) { - const yOffset = node.get_length('-y-offset'); - y = stageY - this.label.height - yOffset; - } else if (opt.DASH_RIGHT) { - const yOffset = Math.floor((itemHeight - labelHeight) / 2); - xOffset = shellVersion46 ? 8 : 4; - - x = stageX - xOffset - this.label.width; - y = Math.clamp(stageY + yOffset, 0, global.stage.height - labelHeight); - } else if (opt.DASH_LEFT) { - const yOffset = Math.floor((itemHeight - labelHeight) / 2); - xOffset = shellVersion46 ? 8 : 4; - - x = stageX + this.width + xOffset; - y = Math.clamp(stageY + yOffset, 0, global.stage.height - labelHeight); - } - - this.label.set_position(x, y); - this.label.ease({ - opacity: 255, - duration: DASH_ITEM_LABEL_SHOW_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - - this.label.set_position(x, y); - this.label.ease({ - opacity: 255, - duration: DASH_ITEM_LABEL_SHOW_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - }, -}; - -const DashCommon = { - _redisplay() { - // After disabling V-Shell queueRedisplay() may call this function - // In that case redirect the call to the current _redisplay() - if (!_moduleEnabled) { - this._redisplay(); - return; - } - - let favorites = AppFavorites.getAppFavorites().getFavoriteMap(); - - let running = this._appSystem.get_running(); - - if (opt.DASH_ISOLATE_WS) { - const currentWs = global.workspace_manager.get_active_workspace(); - running = running.filter(app => { - return app.get_windows().filter(w => w.get_workspace() === currentWs).length; - }); - this._box.get_children().forEach(a => a.child?._updateRunningStyle()); - } - - let children = this._box.get_children().filter(actor => { - return actor.child && - actor.child._delegate && - actor.child._delegate.app; - }); - // Apps currently in the dash - let oldApps = children.map(actor => actor.child._delegate.app); - // Apps supposed to be in the dash - let newApps = []; - - for (let id in favorites) - newApps.push(favorites[id]); - - for (let i = 0; i < running.length; i++) { - let app = running[i]; - if (app.get_id() in favorites) - continue; - 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) { - let oldApp = oldApps.length > oldIndex ? oldApps[oldIndex] : null; - let newApp = newApps.length > newIndex ? newApps[newIndex] : null; - - // No change at oldIndex/newIndex - if (oldApp === newApp) { - oldIndex++; - newIndex++; - continue; - } - - // App removed at oldIndex - if (oldApp && !newApps.includes(oldApp)) { - removedActors.push(children[oldIndex]); - oldIndex++; - continue; - } - - // App added at newIndex - if (newApp && !oldApps.includes(newApp)) { - addedItems.push({ - app: newApp, - item: this._createAppItem(newApp), - pos: newIndex, - }); - newIndex++; - continue; - } - - // App moved - let nextApp = newApps.length > newIndex + 1 - ? newApps[newIndex + 1] : null; - let insertHere = nextApp && nextApp === oldApp; - let alreadyRemoved = removedActors.reduce((result, actor) => { - let removedApp = actor.child._delegate.app; - return result || removedApp === newApp; - }, false); - - if (insertHere || alreadyRemoved) { - let newItem = this._createAppItem(newApp); - addedItems.push({ - app: newApp, - item: newItem, - pos: newIndex + removedActors.length, - }); - newIndex++; - } else { - removedActors.push(children[oldIndex]); - oldIndex++; - } - } - - for (let i = 0; i < addedItems.length; i++) { - this._box.insert_child_at_index( - addedItems[i].item, - addedItems[i].pos); - } - - for (let i = 0; i < removedActors.length; i++) { - let item = removedActors[i]; - - // Don't animate item removal when the overview is transitioning - // or hidden - if (Main.overview.visible && !Main.overview.animationInProgress) - item.animateOutAndDestroy(); - else - item.destroy(); - } - - this._adjustIconSize(); - - // Skip animations on first run when adding the initial set - // of items, to avoid all items zooming in at once - - let animate = this._shownInitially && Main.overview.visible && - !Main.overview.animationInProgress; - - if (!this._shownInitially) - this._shownInitially = true; - - for (let i = 0; i < addedItems.length; i++) - addedItems[i].item.show(animate); - - // Update separator - const nFavorites = Object.keys(favorites).length; - const nIcons = children.length + addedItems.length - removedActors.length; - if (nFavorites > 0 && nFavorites < nIcons) { - // destroy the horizontal separator if it exists. - // this is incredibly janky, but I can't think of a better way atm. - if (this._separator && this._separator.height !== 1) { - this._separator.destroy(); - this._separator = null; - } - - if (!this._separator) { - this._separator = new St.Widget({ - style_class: 'dash-separator', - x_align: Clutter.ActorAlign.CENTER, - y_align: Clutter.ActorAlign.CENTER, - width: opt.DASH_VERTICAL ? this.iconSize : 1, - height: opt.DASH_VERTICAL ? 1 : this.iconSize, - }); - this._box.add_child(this._separator); - } - - // FIXME: separator placement is broken (also in original dash) - let pos = nFavorites + this._animatingPlaceholdersCount; - if (this._dragPlaceholder) - pos++; - this._box.set_child_at_index(this._separator, pos); - } else if (this._separator) { - this._separator.destroy(); - this._separator = null; - } - // Workaround for https://bugzilla.gnome.org/show_bug.cgi?id=692744 - // Without it, StBoxLayout may use a stale size cache - this._box.queue_relayout(); - }, - - _createAppItem(app) { - let appIcon = new Dash.DashIcon(app); - - let indicator = appIcon._dot; - if (opt.DASH_VERTICAL) { - indicator.x_align = opt.DASH_LEFT ? Clutter.ActorAlign.START : Clutter.ActorAlign.END; - indicator.y_align = Clutter.ActorAlign.CENTER; - } else { - indicator.x_align = Clutter.ActorAlign.CENTER; - indicator.y_align = Clutter.ActorAlign.END; - } - - appIcon.connect('menu-state-changed', - (o, opened) => { - this._itemMenuStateChanged(item, opened); - }); - - let item = new Dash.DashItemContainer(); - item.setChild(appIcon); - - // Override default AppIcon label_actor, now the - // accessible_name is set at DashItemContainer.setLabelText - appIcon.label_actor = null; - item.setLabelText(app.get_name()); - - appIcon.icon.setIconSize(this.iconSize); - this._hookUpLabel(item, appIcon); - - return item; - }, - - // use custom BaseIconSizes and add support for custom icons - _adjustIconSize() { - // if a user launches multiple apps at once, this function may be called again before the previous call has finished - // as a result, new icons will not reach their full size, or will be missing, if adding a new icon and changing the dash size due to lack of space at the same time - if (this._adjustingInProgress) - return; - - // 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) - let iconChildren = this._box.get_children().filter(actor => { - return actor.child && - actor.child._delegate && - actor.child._delegate.icon && - !actor.animatingOut; - }); - - // add new custom icons to the list - if (this._showAppsIcon.visible) - iconChildren.push(this._showAppsIcon); - - - // showWindowsIcon and extensionsIcon can be provided by the WSP and ESP extensions - if (this._showWindowsIcon) - iconChildren.push(this._showWindowsIcon); - - if (this._extensionsIcon) - iconChildren.push(this._extensionsIcon); - - - if (!iconChildren.length) - return; - - if (this._maxWidth === -1 || this._maxHeight === -1) - return; - - const dashHorizontal = !opt.DASH_VERTICAL; - - const themeNode = this.get_theme_node(); - const maxAllocation = new Clutter.ActorBox({ - x1: 0, - y1: 0, - x2: dashHorizontal ? this._maxWidth : 42, // not whatever - y2: dashHorizontal ? 42 : this._maxHeight, - }); - - let maxContent = themeNode.get_content_box(maxAllocation); - - let spacing = themeNode.get_length('spacing'); - - let firstButton = iconChildren[0].child; - let firstIcon = firstButton._delegate.icon; - - if (!firstIcon.icon) - return; - - // Enforce valid spacings during the size request - firstIcon.icon.ensure_style(); - const [, , iconWidth, iconHeight] = firstIcon.icon.get_preferred_size(); - const [, , buttonWidth, buttonHeight] = firstButton.get_preferred_size(); - let scaleFactor = St.ThemeContext.get_for_stage(global.stage).scale_factor; - - let maxIconSize = opt.MAX_ICON_SIZE; - if (!maxIconSize) { - maxIconSize = Me.Util.monitorHasLowResolution() - ? 48 - : 64; - } - - let availWidth, availHeight; - if (dashHorizontal) { - availWidth = maxContent.x2 - maxContent.x1; - // Subtract icon padding and box spacing from the available width - availWidth -= iconChildren.length * (buttonWidth - iconWidth) + - (iconChildren.length - 1) * spacing + - 2 * this._background.get_theme_node().get_horizontal_padding(); - - availHeight = this._maxHeight; - availHeight -= this.margin_top + this.margin_bottom; - availHeight -= this._background.get_theme_node().get_vertical_padding(); - availHeight -= themeNode.get_vertical_padding(); - availHeight -= buttonHeight - iconHeight; - - maxIconSize = Math.min(availWidth / iconChildren.length, availHeight, maxIconSize * scaleFactor); - } else { - availWidth = this._maxWidth; - availWidth -= this._background.get_theme_node().get_horizontal_padding(); - availWidth -= themeNode.get_horizontal_padding(); - availWidth -= buttonWidth - iconWidth; - - availHeight = maxContent.y2 - maxContent.y1; - availHeight -= iconChildren.length * (buttonHeight - iconHeight) + - (iconChildren.length - 1) * spacing + - 2 * this._background.get_theme_node().get_vertical_padding(); - - maxIconSize = Math.min(availWidth, availHeight / iconChildren.length, maxIconSize * scaleFactor); - } - - let iconSizes = BaseIconSizes.map(s => s * scaleFactor); - - let newIconSize = BaseIconSizes[0]; - for (let i = 0; i < iconSizes.length; i++) { - if (iconSizes[i] <= maxIconSize) - newIconSize = BaseIconSizes[i]; - } - - if (newIconSize === this.iconSize) - return; - - // set the in-progress state here after all the possible cancels - this._adjustingInProgress = true; - - let oldIconSize = this.iconSize; - this.iconSize = newIconSize; - this.emit('icon-size-changed'); - - let scale = oldIconSize / newIconSize; - for (let i = 0; i < iconChildren.length; i++) { - let icon = iconChildren[i].child._delegate.icon; - - // Set the new size immediately, to keep the icons' sizes - // in sync with this.iconSize - icon.setIconSize(this.iconSize); - - // Don't animate the icon size change when the overview - // is transitioning, not visible or when initially filling - // the dash - if (!Main.overview.visible || Main.overview.animationInProgress || - !this._shownInitially) - continue; - - 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); - - icon.icon.ease({ - width: targetWidth, - height: targetHeight, - duration: Dash.DASH_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - - if (this._separator) { - this._separator.ease({ - width: dashHorizontal ? 1 : this.iconSize, - height: dashHorizontal ? this.iconSize : 1, - duration: Dash.DASH_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - - this._adjustingInProgress = false; - }, - - handleDragOver(source, actor, x, y, _time) { - let app = getAppFromSource(source); - - // Don't allow favoriting of transient apps - if (app === null || app.is_window_backed()) - return DND.DragMotionResult.NO_DROP; - if (!global.settings.is_writable('favorite-apps')) - return DND.DragMotionResult.NO_DROP; - let favorites = AppFavorites.getAppFavorites().getFavorites(); - let numFavorites = favorites.length; - - let favPos = favorites.indexOf(app); - - let children = this._box.get_children(); - let numChildren = children.length; - let boxSize = opt.DASH_VERTICAL ? this._box.height : this._box.width; - - // 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) { - boxSize -= opt.DASH_VERTICAL ? this._dragPlaceholder.height : this._dragPlaceholder.width; - numChildren--; - } - - // Same with the separator - if (this._separator) { - boxSize -= opt.DASH_VERTICAL ? this._separator.height : this._separator.width; - numChildren--; - } - - let pos; - if (this._emptyDropTarget) - pos = 0; // always insert at the start when dash is empty - else if (this.text_direction === Clutter.TextDirection.RTL) - pos = numChildren - Math.floor((opt.DASH_VERTICAL ? y : x) * numChildren / boxSize); - else - pos = Math.floor((opt.DASH_VERTICAL ? y : x) * numChildren / boxSize); - - // 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) { - this._dragPlaceholderPos = pos; - - // Don't allow positioning before or after self - if (favPos !== -1 && (pos === favPos || pos === favPos + 1)) { - this._clearDragPlaceholder(); - return DND.DragMotionResult.CONTINUE; - } - - // 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) { - this._dragPlaceholder.destroy(); - fadeIn = false; - } else { - fadeIn = true; - } - - // this._dragPlaceholder = new Dash.DragPlaceholderItem(); // not exported in 45 - this._dragPlaceholder = new Dash.DashItemContainer(); - this._dragPlaceholder.setChild(new St.Bin({ style_class: 'placeholder' })); - this._dragPlaceholder.child.set_width(this.iconSize / (opt.DASH_VERTICAL ? 2 : 1)); - this._dragPlaceholder.child.set_height(this.iconSize / (opt.DASH_VERTICAL ? 1 : 2)); - this._box.insert_child_at_index( - this._dragPlaceholder, - this._dragPlaceholderPos); - this._dragPlaceholder.show(fadeIn); - } - - if (!this._dragPlaceholder) - return DND.DragMotionResult.NO_DROP; - - let srcIsFavorite = favPos !== -1; - - if (srcIsFavorite) - return DND.DragMotionResult.MOVE_DROP; - - return DND.DragMotionResult.COPY_DROP; - }, -}; - -const AppIconCommon = { - after__init() { - if (this._updateRunningDotStyle) - this._updateRunningDotStyle(); - }, - - _updateRunningDotStyle() { - if (opt.RUNNING_DOT_STYLE) - this._dot.add_style_class_name('app-grid-running-dot-custom'); - else - this._dot.remove_style_class_name('app-grid-running-dot-custom'); - }, - - activate(button) { - const event = Clutter.get_current_event(); - const state = event ? event.get_state() : 0; - const isMiddleButton = button && button === Clutter.BUTTON_MIDDLE; - const isCtrlPressed = Me.Util.isCtrlPressed(state); - const isShiftPressed = Me.Util.isShiftPressed(state); - - const currentWS = global.workspace_manager.get_active_workspace(); - const appRecentWorkspace = this._getAppRecentWorkspace(this.app); - // this feature shouldn't affect search results, dash icons don't have labels, so we use them as a condition - const showWidowsBeforeActivation = opt.DASH_CLICK_ACTION === 1 && !this.icon.label; - - let targetWindowOnCurrentWs = false; - if (opt.DASH_FOLLOW_RECENT_WIN) { - targetWindowOnCurrentWs = appRecentWorkspace === currentWS; - } else { - this.app.get_windows().forEach( - w => { - targetWindowOnCurrentWs = targetWindowOnCurrentWs || (w.get_workspace() === currentWS); - } - ); - } - - const openNewWindow = this.app.can_open_new_window() && - this.app.state === Shell.AppState.RUNNING && - (((isCtrlPressed || isMiddleButton) && !opt.DASH_CLICK_OPEN_NEW_WIN) || - (opt.DASH_CLICK_OPEN_NEW_WIN && !this._selectedMetaWin && !isMiddleButton) || - ((opt.DASH_CLICK_PREFER_WORKSPACE || opt.DASH_ISOLATE_WS) && !targetWindowOnCurrentWs)); - - if ((this.app.state === Shell.AppState.STOPPED || openNewWindow) && !isShiftPressed) - this.animateLaunch(); - - if (openNewWindow) { - this.app.open_new_window(-1); - // if DASH_CLICK_ACTION == "SHOW_WINS_BEFORE", the app has more than one window and has no window on the current workspace, - // don't activate the app immediately, only move the overview to the workspace with the app's recent window - } else if (showWidowsBeforeActivation && !isShiftPressed && this.app.get_n_windows() > 1 && !targetWindowOnCurrentWs/* && !(opt.OVERVIEW_MODE && !opt.WORKSPACE_MODE)*/) { - - Main.wm.actionMoveWorkspace(appRecentWorkspace); - Main.overview.dash.showAppsButton.checked = false; - return; - } else if (this._selectedMetaWin) { - this._selectedMetaWin.activate(global.get_current_time()); - } else if (showWidowsBeforeActivation && opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE && !isShiftPressed && this.app.get_n_windows() > 1) { - // expose windows - Main.overview._overview._controls._thumbnailsBox._activateThumbnailAtPoint(0, 0, global.get_current_time(), true); - return; - } else if (((opt.DASH_SHIFT_CLICK_MV && isShiftPressed) || ((opt.DASH_CLICK_PREFER_WORKSPACE || opt.DASH_ISOLATE_WS) && !openNewWindow)) && this.app.get_windows().length) { - this._moveAppToCurrentWorkspace(); - if (opt.DASH_ISOLATE_WS) { - this.app.activate(); - // hide the overview after the window is re-created - GLib.idle_add(GLib.PRIORITY_LOW, () => Main.overview.hide()); - } - return; - } else if (isShiftPressed) { - return; - } else { - this.app.activate(); - } - - Main.overview.hide(); - }, - - _moveAppToCurrentWorkspace() { - this.app.get_windows().forEach(w => w.change_workspace(global.workspace_manager.get_active_workspace())); - }, - - popupMenu(side = St.Side.LEFT) { - this.setForcedHighlight(true); - this._removeMenuTimeout(); - this.fake_release(); - - if (!this._getWindowsOnCurrentWs) { - this._getWindowsOnCurrentWs = function () { - const winList = []; - this.app.get_windows().forEach(w => { - if (w.get_workspace() === global.workspace_manager.get_active_workspace()) - winList.push(w); - }); - return winList; - }; - - this._windowsOnOtherWs = function () { - return (this.app.get_windows().length - this._getWindowsOnCurrentWs().length) > 0; - }; - } - - if (!this._menu) { - this._menu = new AppMenu.AppMenu(this, side, { - favoritesSection: true, - showSingleWindows: true, - }); - - this._menu.setApp(this.app); - this._openSigId = this._menu.connect('open-state-changed', (menu, isPoppedUp) => { - if (!isPoppedUp) - this._onMenuPoppedDown(); - }); - // Main.overview.connectObject('hiding', - this._hidingSigId = Main.overview.connect('hiding', - () => this._menu.close(), this); - - Main.uiGroup.add_child(this._menu.actor); - this._menuManager.addMenu(this._menu); - } - - // once the menu is created, it stays unchanged and we need to modify our items based on current situation - if (this._addedMenuItems && this._addedMenuItems.length) - this._addedMenuItems.forEach(i => i.destroy()); - - - const popupItems = []; - - const separator = new PopupMenu.PopupSeparatorMenuItem(); - this._menu.addMenuItem(separator); - - if (this.app.get_n_windows()) { - // if (/* opt.APP_MENU_FORCE_QUIT*/true) {} - popupItems.push([_('Force Quit'), () => { - this.app.get_windows()[0].kill(); - }]); - - // if (opt.APP_MENU_CLOSE_WS) {} - const nWin = this._getWindowsOnCurrentWs().length; - if (nWin) { - popupItems.push([_(`Close ${nWin} Windows on Current Workspace`), () => { - const windows = this._getWindowsOnCurrentWs(); - let time = global.get_current_time(); - for (let win of windows) { - // increase time by 1 ms for each window to avoid errors from GS - win.delete(time++); - } - }]); - } - - popupItems.push([_('Move App to Current Workspace ( Shift + Click )'), this._moveAppToCurrentWorkspace]); - // WTMB (Windows Thumbnails) extension required - if (global.windowThumbnails) { - popupItems.push([_('Create Window Thumbnail/PiP'), () => { - global.windowThumbnails?.createThumbnail(this.app.get_windows()[0]); - }]); - } - } - - this._addedMenuItems = []; - this._addedMenuItems.push(separator); - popupItems.forEach(i => { - let item = new PopupMenu.PopupMenuItem(i[0]); - this._menu.addMenuItem(item); - item.connect('activate', i[1].bind(this)); - if (i[1] === this._moveAppToCurrentWorkspace && !this._windowsOnOtherWs()) - item.setSensitive(false); - this._addedMenuItems.push(item); - }); - - this.emit('menu-state-changed', true); - - this._menu.open(BoxPointer.PopupAnimation.FULL); - this._menuManager.ignoreRelease(); - this.emit('sync-tooltip'); - - return false; - }, - - _getWindowApp(metaWin) { - const tracker = Shell.WindowTracker.get_default(); - return tracker.get_window_app(metaWin); - }, - - _getAppLastUsedWindow(app) { - let recentWin; - global.display.get_tab_list(Meta.TabList.NORMAL_ALL, null).forEach(metaWin => { - const winApp = this._getWindowApp(metaWin); - if (!recentWin && winApp === app) - recentWin = metaWin; - }); - return recentWin; - }, - - _getAppRecentWorkspace(app) { - const recentWin = this._getAppLastUsedWindow(app); - if (recentWin) - return recentWin.get_workspace(); - - return null; - }, -}; - -const DashIconCommon = { - after__init() { - if (opt.DASH_ICON_SCROLL && !Me.Util.dashNotDefault()) { - this._scrollConId = this.connect('scroll-event', DashExtensions.onScrollEvent.bind(this)); - this._leaveConId = this.connect('leave-event', DashExtensions.onLeaveEvent.bind(this)); - } - }, - - popupMenu() { - const side = opt.DASH_VERTICAL ? St.Side.LEFT : St.Side.BOTTOM; - AppIconCommon.popupMenu.bind(this)(side); - }, - - _updateRunningDotStyle() { - if (opt.RUNNING_DOT_STYLE) - this._dot.add_style_class_name('app-grid-running-dot-custom'); - else - this._dot.remove_style_class_name('app-grid-running-dot-custom'); - - this._dot.translation_x = 0; - // _updateDotStyle() has been added in GS 46.2 to apply translation_y value from the CSS on style change - if (shellVersion46 && !this._updateDotStyle && !opt.DASH_VERTICAL) - this._dot.translation_y = 8; - - // GS 46.0 (Ubuntu) only - if (opt.DASH_VERTICAL) - this._dot.translationY = 0; - }, - - _updateRunningStyle() { - const currentWs = global.workspace_manager.get_active_workspace(); - const show = opt.DASH_ISOLATE_WS - ? this.app.get_windows().filter(w => w.get_workspace() === currentWs).length - : this.app.state !== Shell.AppState.STOPPED; - - if (show) - this._dot.show(); - else - this._dot.hide(); - }, -}; - -const DashExtensions = { - onScrollEvent(source, event) { - if ((this.app && !opt.DASH_ICON_SCROLL) || (this._isSearchWindowsIcon && !opt.SEARCH_WINDOWS_ICON_SCROLL)) { - if (this._scrollConId) { - this.disconnect(this._scrollConId); - this._scrollConId = 0; - } - if (this._leaveConId) { - this.disconnect(this._leaveConId); - this._leaveConId = 0; - } - return Clutter.EVENT_PROPAGATE; - } - - if (Main.overview._overview.controls._stateAdjustment.value > 1) - return Clutter.EVENT_PROPAGATE; - - let direction = Me.Util.getScrollDirection(event); - if (direction === Clutter.ScrollDirection.UP) - direction = 1; - else if (direction === Clutter.ScrollDirection.DOWN) - direction = -1; - else - return Clutter.EVENT_STOP; - - // avoid uncontrollable switching if smooth scroll wheel or trackpad is used - if (this._lastScroll && Date.now() - this._lastScroll < 160) - return Clutter.EVENT_STOP; - - this._lastScroll = Date.now(); - - DashExtensions.switchWindow.bind(this)(direction); - return Clutter.EVENT_STOP; - }, - - onLeaveEvent() { - if (!this._selectedMetaWin || this.has_pointer || this.toggleButton?.has_pointer) - return; - - this._selectedPreview._activateSelected = false; - this._selectedMetaWin = null; - this._scrolledWindows = null; - DashExtensions.showWindowPreview.bind(this)(null); - }, - - - switchWindow(direction) { - if (!this._scrolledWindows) { - this._initialSelection = true; - // source is app icon - if (this.app) { - this._scrolledWindows = this.app.get_windows(); - if (opt.DASH_ISOLATE_WS) { - const currentWs = global.workspaceManager.get_active_workspace(); - this._scrolledWindows = this._scrolledWindows.filter(w => w.get_workspace() === currentWs); - } - - const wsList = []; - this._scrolledWindows.forEach(w => { - const ws = w.get_workspace(); - if (!wsList.includes(ws)) - wsList.push(ws); - }); - - // sort windows by workspaces in MRU order - this._scrolledWindows.sort((a, b) => wsList.indexOf(a.get_workspace()) > wsList.indexOf(b.get_workspace())); - // source is Search Windows icon - } else if (this._isSearchWindowsIcon) { - if (opt.SEARCH_WINDOWS_ICON_SCROLL === 1) // all windows - this._scrolledWindows = Me.Util.getWindows(null); - else - this._scrolledWindows = Me.Util.getWindows(global.workspace_manager.get_active_workspace()); - } - } - - let windows = this._scrolledWindows; - - if (!windows.length) - return; - - // if window selection is in the process, the previewed window must be the current one - let currentWin = this._selectedMetaWin ? this._selectedMetaWin : windows[0]; - - const currentIdx = windows.indexOf(currentWin); - let targetIdx = currentIdx; - // const focusWindow = Me.Util.getWindows(null)[0]; // incompatible 45 - const focusWindow = Me.Util.getWindows(null)[0]; - const appFocused = this._scrolledWindows[0] === focusWindow && this._scrolledWindows[0].get_workspace() === global.workspace_manager.get_active_workspace(); - // only if the app has focus, immediately switch to the previous window - // otherwise just set the current window above others - if (!this._initialSelection || appFocused) - targetIdx += direction; - else - this._initialSelection = false; - - if (targetIdx > windows.length - 1) - targetIdx = 0; - else if (targetIdx < 0) - targetIdx = windows.length - 1; - - const metaWin = windows[targetIdx]; - DashExtensions.showWindowPreview.bind(this)(metaWin); - this._selectedMetaWin = metaWin; - }, - - showWindowPreview(metaWin) { - const views = Main.overview._overview.controls._workspacesDisplay._workspacesViews; - const viewsIter = [views[0]]; - // secondary monitors use different structure - views.forEach(v => { - if (v._workspacesView) - viewsIter.push(v._workspacesView); - }); - - viewsIter.forEach(view => { - // if workspaces are on primary monitor only - if (!view || !view._workspaces) - return; - - view._workspaces.forEach(ws => { - ws._windows.forEach(windowPreview => { - // metaWin === null resets opacity - let opacity = metaWin ? 50 : 255; - windowPreview._activateSelected = false; - - // minimized windows are invisible if windows are not exposed (WORKSPACE_MODE === 0) - if (!windowPreview.opacity) - windowPreview.opacity = 255; - - // app windows set to lower opacity, so they can be recognized - if (this._scrolledWindows && this._scrolledWindows.includes(windowPreview.metaWindow)) { - if (opt.DASH_ICON_SCROLL === 2) - opacity = 254; - } - if (windowPreview.metaWindow === metaWin) { - if (metaWin && metaWin.get_workspace() !== global.workspace_manager.get_active_workspace()) { - Main.wm.actionMoveWorkspace(metaWin.get_workspace()); - if (_timeouts.wsSwitcherAnimation) - GLib.source_remove(_timeouts.wsSwitcherAnimation); - // setting window preview above siblings before workspace switcher animation has no effect - // we need to set the window above after the ws preview become visible on the screen - // the default switcher animation time is 250, 200 ms delay should be enough - _timeouts.wsSwitcherAnimation = GLib.timeout_add(0, 200 * St.Settings.get().slow_down_factor, () => { - windowPreview.get_parent().set_child_above_sibling(windowPreview, null); - _timeouts.wsSwitcherAnimation = 0; - return GLib.SOURCE_REMOVE; - }); - } else { - windowPreview.get_parent().set_child_above_sibling(windowPreview, null); - } - - opacity = 255; - this._selectedPreview = windowPreview; - windowPreview._activateSelected = true; - } - - // if windows are exposed, highlight selected using opacity - if ((opt.OVERVIEW_MODE && opt.WORKSPACE_MODE) || !opt.OVERVIEW_MODE) { - if (metaWin && opacity === 255) - windowPreview.showOverlay(true); - else - windowPreview.hideOverlay(true); - windowPreview.ease({ - duration: 200, - opacity, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - }); - }); - }); - }, -}; - -const AppMenuCommon = { - _updateWindowsSection() { - if (global.compositor) { - if (this._updateWindowsLaterId) { - const laters = global.compositor.get_laters(); - laters.remove(this._updateWindowsLaterId); - } - } else if (this._updateWindowsLaterId) { - Meta.later_remove(this._updateWindowsLaterId); - } - - this._updateWindowsLaterId = 0; - - this._windowSection.removeAll(); - this._openWindowsHeader.hide(); - - if (!this._app) - return; - - const minWindows = this._showSingleWindows ? 1 : 2; - const currentWs = global.workspaceManager.get_active_workspace(); - const isolateWs = opt.DASH_ISOLATE_WS && !Main.overview.dash.showAppsButton.checked; - const windows = this._app.get_windows().filter(w => !w.skip_taskbar && (isolateWs ? w.get_workspace() === currentWs : true)); - if (windows.length < minWindows) - return; - - this._openWindowsHeader.show(); - - windows.forEach(window => { - const title = window.title || this._app.get_name(); - const item = this._windowSection.addAction(title, event => { - Main.activateWindow(window, event.get_time()); - }); - window.connectObject('notify::title', () => { - item.label.text = window.title || this._app.get_name(); - }, item); - }); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/iconGrid.js b/extensions/46/vertical-workspaces/lib/iconGrid.js deleted file mode 100644 index f0c6b18..0000000 --- a/extensions/46/vertical-workspaces/lib/iconGrid.js +++ /dev/null @@ -1,429 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * iconGrid.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import St from 'gi://St'; -import GLib from 'gi://GLib'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as IconGrid from 'resource:///org/gnome/shell/ui/iconGrid.js'; - -let Me; -let opt; - -// added sizes for better scaling -export const IconSize = { - LARGEST: 256, - 224: 224, - 208: 208, - 192: 192, - 176: 176, - 160: 160, - 144: 144, - 128: 128, - 112: 112, - LARGE: 96, - 80: 80, - 64: 64, - TINY: 48, -}; - -export const IconGridModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('appDisplayModule'); - // if notifications are enabled no override is needed - reset = reset || !this.moduleEnabled; - - // don't touch original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('IconGrid', IconGrid.IconGrid.prototype, IconGridCommon); - this._overrides.addOverride('IconGridLayout', IconGrid.IconGridLayout.prototype, IconGridLayoutCommon); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - } -}; - -const IconGridCommon = { - getItemsAtPage(page) { - if (page < 0 || page >= this.nPages) - return []; - // throw new Error(`Page ${page} does not exist at IconGrid`); - - const layoutManager = this.layout_manager; - return layoutManager.getItemsAtPage(page); - }, - - _shouldUpdateGrid(width, height) { - if (this.layoutManager._isFolder) - return false; - else if (this._currentMode === -1) - return true; - - // Update if page size changed - // Page dimensions may change within a small range - const range = 5; - return (Math.abs(width - (this._gridForWidth ?? 0)) > range) || - (Math.abs(height - (this._gridForHeight ?? 0)) > range); - }, - - _findBestModeForSize(width, height) { - // this function is for main grid only, folder grid calculation is in appDisplay.AppFolderDialog class - if (!this._shouldUpdateGrid(width, height)) - return; - - this._gridForWidth = width; - this._gridForHeight = height; - - this._updateDefaultIconSize(); - const { pagePadding } = this.layout_manager; - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const itemPadding = 55; - - // pagePadding is already affected by the scaleFactor - width -= pagePadding.left + pagePadding.right; - height -= pagePadding.top + pagePadding.bottom; - - // Sync with _findBestIconSize() - this.layoutManager._gridSizeChanged = true; - this.layoutManager._gridWidth = width; - this.layoutManager._gridHeight = height; - - // All widgets are affected by the scaleFactor so we need to apply it also on the page size - width /= scaleFactor; - height /= scaleFactor; - - const spacing = opt.APP_GRID_SPACING; - const iconSize = opt.APP_GRID_ICON_SIZE > 0 ? opt.APP_GRID_ICON_SIZE : opt.APP_GRID_ICON_SIZE_DEFAULT; - const itemSize = iconSize + itemPadding; - let columns = opt.APP_GRID_COLUMNS; - let rows = opt.APP_GRID_ROWS; - // 0 means adaptive size - let unusedSpaceH = -1; - if (!columns) { - // calculate #columns + 1 without spacing - columns = Math.floor(width / itemSize) + 1; - // check if columns with spacing fits the available width - // and reduce the number until it fits - while (unusedSpaceH < 0) { - columns -= 1; - unusedSpaceH = width - columns * itemSize - (columns - 1) * spacing; - } - } - let unusedSpaceV = -1; - if (!rows) { - rows = Math.floor(height / itemSize) + 1; - while (unusedSpaceV < 0) { - rows -= 1; - unusedSpaceV = height - rows * itemSize - ((rows - 1) * spacing); - } - } - - this._gridModes = [{ columns, rows }]; - this._currentMode = -1; - this._setGridMode(0); - this.layoutManager.updateIconSize(); - // Call _redisplay() from timeout to avoid allocation errors - GLib.idle_add(GLib.PRIORITY_LOW, () => - Main.overview._overview.controls.appDisplay._redisplay() - ); - }, - - _updateDefaultIconSize() { - // Reduce default icon size for low resolution screens and high screen scales - if (Me.Util.monitorHasLowResolution()) { - opt.APP_GRID_ICON_SIZE_DEFAULT = opt.APP_GRID_ACTIVE_PREVIEW && !opt.APP_GRID_USAGE ? 128 : 64; - opt.APP_GRID_FOLDER_ICON_SIZE_DEFAULT = 64; - } else { - opt.APP_GRID_ICON_SIZE_DEFAULT = opt.APP_GRID_ACTIVE_PREVIEW && !opt.APP_GRID_USAGE ? 192 : 96; - } - }, - - // Workaround for the upstream bug - // https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/5753 - // https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/5240 - // https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/6892 - // The appGridLayout._currentPage is not updated when the page is changed in the grid - // For example, when user navigates app icons using a keyboard - // Related issues open on GNOME's gitlab: - after_goToPage() { - if (this._delegate._appGridLayout._currentPage !== this._currentPage) - this._delegate._appGridLayout.goToPage(this._currentPage); - }, - - // Workaround for the upstream bug - // https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/7700 - // Return INVALID target if x or y is out of the grid view to prevent pages[page] undefined error (horizontal orientation only) - getDropTarget(x, y) { - if (x < 0 || y < 0) - return [0, 0, 0]; // [0, 0, DragLocation.INVALID] - const layoutManager = this.layout_manager; - return layoutManager.getDropTarget(x, y, this._currentPage); - }, -}; - -const IconGridLayoutCommon = { - _findBestIconSize() { - if (this.fixedIconSize !== -1) - return this.fixedIconSize; - - if (!this._isFolder && !this._gridSizeChanged) - return this._iconSize; - this._gridSizeChanged = false; - - - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const nColumns = this.columnsPerPage; - const nRows = this.rowsPerPage; - - // If grid is not defined, return default icon size - if (nColumns < 1 && nRows < 1) { - return this._isFolder - ? opt.APP_GRID_FOLDER_ICON_SIZE_DEFAULT - : opt.APP_GRID_ICON_SIZE_DEFAULT; - } - - const spacing = this._isFolder - ? opt.APP_GRID_FOLDER_SPACING - : opt.APP_GRID_SPACING; - - const columnSpacingPerPage = spacing * (nColumns - 1); - const rowSpacingPerPage = spacing * (nRows - 1); - const itemPadding = 55; - - const width = (this._gridWidth ? this._gridWidth : this._pageWidth) / scaleFactor; - let height = (this._gridHeight ? this._gridHeight : this._pageHeight) / scaleFactor; - - if (!width || !height) - return opt.APP_GRID_ICON_SIZE_DEFAULT; - - const [firstItem] = this._container; - - let iconSizes = Object.values(IconSize).sort((a, b) => b - a); - // Limit max icon size for folders and fully adaptive folder grids, the whole range is for the main grid with active folders - if (this._isFolder && opt.APP_GRID_FOLDER_ICON_SIZE < 0) - iconSizes = iconSizes.slice(iconSizes.indexOf(opt.APP_GRID_FOLDER_ICON_SIZE_DEFAULT), -1); - else if (this._isFolder) - iconSizes = iconSizes.slice(iconSizes.indexOf(IconSize.LARGE), -1); - else if (opt.APP_GRID_ICON_SIZE < 0) - iconSizes = iconSizes.slice(iconSizes.indexOf(opt.APP_GRID_ICON_SIZE_DEFAULT), -1); - - let sizeInvalid = false; - for (const size of iconSizes) { - let usedWidth, usedHeight; - - if (firstItem) { - firstItem.icon.setIconSize(size); - const [firstItemWidth] = firstItem.get_preferred_size(); - - const itemSize = firstItemWidth / scaleFactor; - if (itemSize < size) - sizeInvalid = true; - - usedWidth = itemSize * nColumns; - usedHeight = itemSize * nRows; - } - - if (!firstItem || sizeInvalid) { - usedWidth = (size + itemPadding) * nColumns; - usedHeight = (size + itemPadding) * nRows; - } - const emptyHSpace = - width - usedWidth - columnSpacingPerPage; - const emptyVSpace = - height - usedHeight - rowSpacingPerPage; - - if (emptyHSpace >= 0 && emptyVSpace >= 0) - return size; - } - - return IconSize.TINY; - }, - - removeItem(item) { - if (!this._items.has(item)) { - console.error(`iconGrid: Item ${item} is not part of the IconGridLayout`); - return; - // throw new Error(`Item ${item} is not part of the IconGridLayout`); - } - - if (!this._container) - return; - - this._shouldEaseItems = true; - - this._container.remove_child(item); - this._removeItemData(item); - }, - - addItem(item, page = -1, index = -1) { - if (this._items.has(item)) { - console.error(`iconGrid: Item ${item} already added to IconGridLayout`); - return; - // throw new Error(`Item ${item} already added to IconGridLayout`); - } - - if (page > this._pages.length) { - console.error(`iconGrid: Cannot add ${item} to page ${page}`); - page = -1; - index = -1; - // throw new Error(`Cannot add ${item} to page ${page}`); - } - - if (!this._container) - return; - - if (page !== -1 && index === -1) - page = this._findBestPageToAppend(page); - - this._shouldEaseItems = true; - - if (!this._container.get_children().includes(item)) - this._container.add_child(item); - this._addItemToPage(item, page, index); - }, - - moveItem(item, newPage, newPosition) { - if (!this._items.has(item)) { - console.error(`iconGrid: Item ${item} is not part of the IconGridLayout`); - return; - // throw new Error(`Item ${item} is not part of the IconGridLayout`); - } - - this._shouldEaseItems = true; - - this._removeItemData(item); - - if (newPage !== -1 && newPosition === -1) - newPage = this._findBestPageToAppend(newPage); - this._addItemToPage(item, newPage, newPosition); - }, - - _addItemToPage(item, pageIndex, index) { - // Ensure we have at least one page - if (this._pages.length === 0) - this._appendPage(); - - // Append a new page if necessary - if (pageIndex === this._pages.length) - this._appendPage(); - - if (pageIndex >= this._pages.length) { - pageIndex = -1; - index = -1; - } - - if (pageIndex === -1) - pageIndex = this._pages.length - 1; - - if (index === -1) - index = this._pages[pageIndex].children.length; - - this._items.set(item, { - actor: item, - pageIndex, - destroyId: item.connect('destroy', () => this._removeItemData(item)), - visibleId: item.connect('notify::visible', () => { - const itemData = this._items.get(item); - - this._updateVisibleChildrenForPage(itemData.pageIndex); - - if (item.visible) - this._relocateSurplusItems(itemData.pageIndex); - else if (!this.allowIncompletePages) - this._fillItemVacancies(itemData.pageIndex); - }), - queueRelayoutId: item.connect('queue-relayout', () => { - this._childrenMaxSize = -1; - }), - }); - - item.icon.setIconSize(this._iconSize); - this._pages[pageIndex].children.splice(index, 0, item); - this._updateVisibleChildrenForPage(pageIndex); - this._relocateSurplusItems(pageIndex); - }, - - _relocateSurplusItems(pageIndex) { - // Avoid recursion during relocations in _redisplay() - if (this._skipRelocateSurplusItems) - return; - - const visiblePageItems = this._pages[pageIndex].visibleChildren; - const itemsPerPage = this.columnsPerPage * this.rowsPerPage; - - // No overflow - if (visiblePageItems.length <= itemsPerPage) - return; - - const nExtraItems = visiblePageItems.length - itemsPerPage; - for (let i = 0; i < nExtraItems; i++) { - const overflowIndex = visiblePageItems.length - i - 1; - const overflowItem = visiblePageItems[overflowIndex]; - - this._removeItemData(overflowItem); - this._addItemToPage(overflowItem, pageIndex + 1, 0); - } - }, - - _findBestPageToAppend(startPage) { - const itemsPerPage = this.columnsPerPage * this.rowsPerPage; - - for (let i = startPage; i < this._pages.length; i++) { - const visibleItems = this._pages[i].visibleChildren; - - if (visibleItems.length < itemsPerPage) - return i; - } - - return this._pages.length; - }, - - updateIconSize() { - const iconSize = this._findBestIconSize(); - if (this._iconSize !== iconSize) { - this._iconSize = iconSize; - - for (const child of this._container) - child.icon.setIconSize(iconSize); - - this.notify('icon-size'); - } - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/layout.js b/extensions/46/vertical-workspaces/lib/layout.js deleted file mode 100644 index 807f9e0..0000000 --- a/extensions/46/vertical-workspaces/lib/layout.js +++ /dev/null @@ -1,473 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * layout.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import GLib from 'gi://GLib'; -import Meta from 'gi://Meta'; -import Gio from 'gi://Gio'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Layout from 'resource:///org/gnome/shell/ui/layout.js'; - -let Me; -let opt; -let _timeouts; - -export const LayoutModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - _timeouts = {}; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - this._originalUpdateHotCorners = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this._removeTimeouts(); - - this.moduleEnabled = opt.get('layoutModule'); - const conflict = Me.Util.getEnabledExtensions('custom-hot-corners').length || - Me.Util.getEnabledExtensions('dash-to-panel').length; - - if (conflict && !reset) - console.warn(`[${Me.metadata.name}] Warning: "Layout" module disabled due to potential conflict with another extension`); - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' LayoutModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - _timeouts = {}; - - this._overrides.addOverride('LayoutManager', Main.layoutManager, LayoutManagerCommon); - this._overrides.addOverride('HotCorner', Layout.HotCorner.prototype, HotCornerCommon); - - Main.layoutManager._updatePanelBarrier(); - Main.layoutManager._updateHotCorners(); - - if (!this._hotCornersEnabledConId) { - this._interfaceSettings = new Gio.Settings({ - schema_id: 'org.gnome.desktop.interface', - }); - this._hotCornersEnabledConId = this._interfaceSettings.connect('changed::enable-hot-corners', - () => Main.layoutManager._updateHotCorners()); - } - - console.debug(' LayoutModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - Main.layoutManager._updateHotCorners(); - - if (this._hotCornersEnabledConId) { - this._interfaceSettings.disconnect(this._hotCornersEnabledConId); - this._hotCornersEnabledConId = 0; - this._interfaceSettings = null; - } - - console.debug(' LayoutModule - Disabled'); - } - - _removeTimeouts() { - if (_timeouts) { - Object.values(_timeouts).forEach(t => { - if (t) - GLib.source_remove(t); - }); - _timeouts = null; - } - } -}; - -const LayoutManagerCommon = { - _updatePanelBarrier() { - if (this._rightPanelBarrier) { - this._rightPanelBarrier.destroy(); - this._rightPanelBarrier = null; - } - - if (this._leftPanelBarrier) { - this._leftPanelBarrier.destroy(); - this._leftPanelBarrier = null; - } - - if (!this.primaryMonitor || !opt || Me.Util.getEnabledExtensions('hidetopbar')) - return; - - if (this.panelBox.height) { - const backend = !!Meta.Barrier.prototype.backend; - let params = {}; - if (backend) - params['backend'] = global.backend; - else - params['display'] = global.display; - - let primary = this.primaryMonitor; - if ([0, 1, 3].includes(opt.HOT_CORNER_POSITION)) { - params = Object.assign({}, params, { - x1: primary.x + primary.width, y1: this.panelBox.allocation.y1, - x2: primary.x + primary.width, y2: this.panelBox.allocation.y2, - directions: Meta.BarrierDirection.NEGATIVE_X, - }); - this._rightPanelBarrier = new Meta.Barrier(params); - } - - if ([2, 4].includes(opt.HOT_CORNER_POSITION)) { - params = Object.assign({}, params, { - x1: primary.x, y1: this.panelBox.allocation.y1, - x2: primary.x, y2: this.panelBox.allocation.y2, - directions: Meta.BarrierDirection.POSITIVE_X, - }); - this._leftPanelBarrier = new Meta.Barrier(params); - } - } - }, - - _updateHotCorners() { - // avoid errors if called from foreign override - if (!opt) - return; - - // destroy old hot corners - this.hotCorners.forEach(corner => corner?.destroy()); - this.hotCorners = []; - - if (!this._interfaceSettings.get_boolean('enable-hot-corners')) { - this.emit('hot-corners-changed'); - return; - } - - let size = this.panelBox.height ? this.panelBox.height : 27; - - // position 0 - default, 1-TL, 2-TR, 3-BL, 4-BR - const position = opt.HOT_CORNER_POSITION; - - // build new hot corners - for (let i = 0; i < this.monitors.length; i++) { - let monitor = this.monitors[i]; - let cornerX, cornerY; - - if (position === 0) { - cornerX = this._rtl ? monitor.x + monitor.width : monitor.x; - cornerY = monitor.y; - } else if (position === 1) { - cornerX = monitor.x; - cornerY = monitor.y; - } else if (position === 2) { - cornerX = monitor.x + monitor.width; - cornerY = monitor.y; - } else if (position === 3) { - cornerX = monitor.x; - cornerY = monitor.y + monitor.height; - } else { - cornerX = monitor.x + monitor.width; - cornerY = monitor.y + monitor.height; - } - - let haveCorner = true; - - if (i !== this.primaryIndex) { - // Check if we have a top left (right for RTL) corner. - // I.e. if there is no monitor directly above or to the left(right) - let besideX = this._rtl ? monitor.x + 1 : cornerX - 1; - let besideY = cornerY; - let aboveX = cornerX; - let aboveY = cornerY - 1; - - for (let j = 0; j < this.monitors.length; j++) { - if (i === j) - continue; - let otherMonitor = this.monitors[j]; - if (besideX >= otherMonitor.x && - besideX < otherMonitor.x + otherMonitor.width && - besideY >= otherMonitor.y && - besideY < otherMonitor.y + otherMonitor.height) { - haveCorner = false; - break; - } - if (aboveX >= otherMonitor.x && - aboveX < otherMonitor.x + otherMonitor.width && - aboveY >= otherMonitor.y && - aboveY < otherMonitor.y + otherMonitor.height) { - haveCorner = false; - break; - } - } - } - - if (haveCorner) { - let corner = new Layout.HotCorner(this, monitor, cornerX, cornerY); - corner.setBarrierSize(size, false); - this.hotCorners.push(corner); - } else { - this.hotCorners.push(null); - } - } - - this.emit('hot-corners-changed'); - }, -}; - -const HotCornerCommon = { - after__init() { - let angle = 0; - switch (opt.HOT_CORNER_POSITION) { - case 2: - angle = 90; - break; - case 3: - angle = 270; - break; - case 4: - angle = 180; - break; - } - - this._ripples._ripple1.rotation_angle_z = angle; - this._ripples._ripple2.rotation_angle_z = angle; - this._ripples._ripple3.rotation_angle_z = angle; - }, - - setBarrierSize(size, notMyCall = true) { - // ignore calls from the original _updateHotCorners() callback to avoid building barriers outside screen - if (notMyCall && size > 0) - return; - - if (this._verticalBarrier) { - this._pressureBarrier.removeBarrier(this._verticalBarrier); - this._verticalBarrier.destroy(); - this._verticalBarrier = null; - } - - if (this._horizontalBarrier) { - this._pressureBarrier.removeBarrier(this._horizontalBarrier); - this._horizontalBarrier.destroy(); - this._horizontalBarrier = null; - } - - if (size > 0) { - const primaryMonitor = global.display.get_primary_monitor(); - const monitor = this._monitor; - const extendV = opt && opt.HOT_CORNER_ACTION && opt.HOT_CORNER_EDGE && opt.DASH_VERTICAL && monitor.index === primaryMonitor; - const extendH = opt && opt.HOT_CORNER_ACTION && opt.HOT_CORNER_EDGE && !opt.DASH_VERTICAL && monitor.index === primaryMonitor; - - const backend = !!Meta.Barrier.prototype.backend; - let params = {}; - if (backend) - params['backend'] = global.backend; - else - params['display'] = global.display; - - if (opt.HOT_CORNER_POSITION <= 1) { - params = Object.assign({}, params, { - x1: this._x, x2: this._x, - y1: this._y, y2: this._y + (extendV ? monitor.height : size), - directions: Meta.BarrierDirection.POSITIVE_X, - }); - this._verticalBarrier = new Meta.Barrier(params); - params = Object.assign({}, params, { - x1: this._x, x2: this._x + (extendH ? monitor.width : size), - y1: this._y, y2: this._y, - directions: Meta.BarrierDirection.POSITIVE_Y, - }); - this._horizontalBarrier = new Meta.Barrier(params); - } else if (opt.HOT_CORNER_POSITION === 2) { - params = Object.assign({}, params, { - x1: this._x, x2: this._x, - y1: this._y, y2: this._y + (extendV ? monitor.height : size), - directions: Meta.BarrierDirection.NEGATIVE_X, - }); - this._verticalBarrier = new Meta.Barrier(params); - params = Object.assign({}, params, { - x1: this._x - size, x2: this._x, - y1: this._y, y2: this._y, - directions: Meta.BarrierDirection.POSITIVE_Y, - }); - this._horizontalBarrier = new Meta.Barrier(params); - } else if (opt.HOT_CORNER_POSITION === 3) { - params = Object.assign({}, params, { - x1: this._x, x2: this._x, - y1: this._y, y2: this._y - size, - directions: Meta.BarrierDirection.POSITIVE_X, - }); - this._verticalBarrier = new Meta.Barrier(params); - params = Object.assign({}, params, { - x1: this._x, x2: this._x + (extendH ? monitor.width : size), - y1: this._y, y2: this._y, - directions: Meta.BarrierDirection.NEGATIVE_Y, - }); - this._horizontalBarrier = new Meta.Barrier(params); - } else if (opt.HOT_CORNER_POSITION === 4) { - params = Object.assign({}, params, { - x1: this._x, x2: this._x, - y1: this._y, y2: this._y - size, - directions: Meta.BarrierDirection.NEGATIVE_X, - }); - this._verticalBarrier = new Meta.Barrier(params); - params = Object.assign({}, params, { - x1: this._x, x2: this._x - size, - y1: this._y, y2: this._y, - directions: Meta.BarrierDirection.NEGATIVE_Y, - }); - this._horizontalBarrier = new Meta.Barrier(params); - } - - this._pressureBarrier.addBarrier(this._verticalBarrier); - this._pressureBarrier.addBarrier(this._horizontalBarrier); - } - }, - - _toggleOverview() { - if (!opt.HOT_CORNER_ACTION || (!opt.HOT_CORNER_FULLSCREEN && this._monitor.inFullscreen && !Main.overview.visible)) - return; - - if (Main.overview.shouldToggleByCornerOrButton()) { - if (Main.overview._shown) { - this._toggleWindowPicker(true); - } else if ((opt.HOT_CORNER_ACTION === 2 && !Me.Util.isCtrlPressed()) || ([3, 4, 5, 6].includes(opt.HOT_CORNER_ACTION) && Me.Util.isCtrlPressed())) { - // Default overview - opt.OVERVIEW_MODE = 0; - opt.OVERVIEW_MODE2 = false; - opt.WORKSPACE_MODE = 1; - this._toggleWindowPicker(true, true); - } else if (opt.HOT_CORNER_ACTION === 1) { - Main.overview.resetOverviewMode(); - this._toggleWindowPicker(true, true); - } else if ((opt.HOT_CORNER_ACTION === 3 && !Me.Util.isCtrlPressed()) || (opt.HOT_CORNER_ACTION === 2 && Me.Util.isCtrlPressed()) || (opt.HOT_CORNER_ACTION === 6 && Me.Util.isCtrlPressed())) { - // Applications - this._toggleApplications(true); - } else if (opt.HOT_CORNER_ACTION === 4 && !Me.Util.isCtrlPressed()) { - // Overview - static ws preview - opt.OVERVIEW_MODE = 1; - opt.OVERVIEW_MODE2 = false; - opt.WORKSPACE_MODE = 0; - this._toggleWindowPicker(true, true); - } else if (opt.HOT_CORNER_ACTION === 5 && !Me.Util.isCtrlPressed()) { - // Overview - static ws - opt.OVERVIEW_MODE = 2; - opt.OVERVIEW_MODE2 = true; - opt.WORKSPACE_MODE = 0; - this._toggleWindowPicker(true, true); - } else if (opt.HOT_CORNER_ACTION === 6 && !Me.Util.isCtrlPressed()) { - // Window search provider - opt.OVERVIEW_MODE = 2; - opt.OVERVIEW_MODE2 = true; - opt.WORKSPACE_MODE = 0; - this._toggleWindowSearchProvider(); - } - if (opt.HOT_CORNER_RIPPLES && Main.overview.animationInProgress) - this._ripples.playAnimation(this._x, this._y); - } - }, - - _toggleWindowPicker(leaveOverview = false, customOverviewMode = false) { - if (Main.overview._shown && (leaveOverview || !Main.overview.dash.showAppsButton.checked)) { - Main.overview.hide(); - } else if (Main.overview.dash.showAppsButton.checked) { - Main.overview.dash.showAppsButton.checked = false; - } else { - const focusWindow = global.display.get_focus_window(); - // at least GS 42 is unable to show overview in X11 session if VirtualBox Machine window grabbed keyboard - if (!Meta.is_wayland_compositor() && focusWindow && focusWindow.wm_class.includes('VirtualBox Machine')) { - // following should help when windowed VBox Machine has focus. - global.stage.set_key_focus(Main.panel); - // key focus doesn't take the effect immediately, we must wait for it - // still looking for better solution! - _timeouts.releaseKeyboardTimeoutId = GLib.timeout_add( - GLib.PRIORITY_DEFAULT, - // delay cannot be too short - 200, - () => { - Main.overview.show(1, customOverviewMode); - - _timeouts.releaseKeyboardTimeoutId = 0; - return GLib.SOURCE_REMOVE; - } - ); - } else { - Main.overview.show(1, customOverviewMode); - } - } - }, - - _toggleApplications(leaveOverview = false) { - if ((leaveOverview && Main.overview._shown) || Main.overview.dash.showAppsButton.checked) { - Main.overview.hide(); - } else { - const focusWindow = global.display.get_focus_window(); - // at least GS 42 is unable to show overview in X11 session if VirtualBox Machine window grabbed keyboard - if (!Meta.is_wayland_compositor() && focusWindow && focusWindow.wm_class.includes('VirtualBox Machine')) { - // following should help when windowed VBox Machine has focus. - global.stage.set_key_focus(Main.panel); - // key focus doesn't take the effect immediately, we must wait for it - // still looking for better solution! - _timeouts.releaseKeyboardTimeoutId = GLib.timeout_add( - GLib.PRIORITY_DEFAULT, - // delay cannot be too short - 200, - () => { - Main.overview.show(2); - - _timeouts.releaseKeyboardTimeoutId = 0; - return GLib.SOURCE_REMOVE; - } - ); - } else if (Main.overview._shown) { - Main.overview.dash.showAppsButton.checked = true; - } else { - Main.overview.show(2); // 2 for App Grid - } - } - }, - - _toggleWindowSearchProvider() { - if (!Main.overview.searchController._searchActive) { - opt.OVERVIEW_MODE = 2; - opt.OVERVIEW_MODE2 = true; - opt.WORKSPACE_MODE = 0; - this._toggleWindowPicker(false, true); - const prefix = Me.WSP_PREFIX; - const position = prefix.length; - const searchEntry = Main.overview.searchEntry; - searchEntry.set_text(prefix); - // searchEntry.grab_key_focus(); - searchEntry.get_first_child().set_cursor_position(position); - searchEntry.get_first_child().set_selection(position, position); - } else { - // Main.overview.searchEntry.text = ''; - Main.overview.hide(); - } - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/messageTray.js b/extensions/46/vertical-workspaces/lib/messageTray.js deleted file mode 100644 index 28d6b1d..0000000 --- a/extensions/46/vertical-workspaces/lib/messageTray.js +++ /dev/null @@ -1,91 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * messageTray.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; - -let Me; -let opt; - -export const MessageTrayModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('messageTrayModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' MessageTrayModule - Keeping untouched'); - } - - _activateModule() { - this._setNotificationPosition(opt.NOTIFICATION_POSITION); - - console.debug(' MessageTrayModule - Activated'); - } - - _disableModule() { - this._setNotificationPosition(1); - - console.debug(' MessageTrayModule - Disabled'); - } - - _setNotificationPosition(position) { - switch (position) { - case 0: - Main.messageTray._bannerBin.x_align = Clutter.ActorAlign.START; - Main.messageTray._bannerBin.y_align = Clutter.ActorAlign.START; - break; - case 1: - Main.messageTray._bannerBin.x_align = Clutter.ActorAlign.CENTER; - Main.messageTray._bannerBin.y_align = Clutter.ActorAlign.START; - break; - case 2: - Main.messageTray._bannerBin.x_align = Clutter.ActorAlign.END; - Main.messageTray._bannerBin.y_align = Clutter.ActorAlign.START; - break; - case 3: - Main.messageTray._bannerBin.x_align = Clutter.ActorAlign.START; - Main.messageTray._bannerBin.y_align = Clutter.ActorAlign.END; - break; - case 4: - Main.messageTray._bannerBin.x_align = Clutter.ActorAlign.CENTER; - Main.messageTray._bannerBin.y_align = Clutter.ActorAlign.END; - break; - case 5: - Main.messageTray._bannerBin.x_align = Clutter.ActorAlign.END; - Main.messageTray._bannerBin.y_align = Clutter.ActorAlign.END; - break; - } - } -}; diff --git a/extensions/46/vertical-workspaces/lib/optionsFactory.js b/extensions/46/vertical-workspaces/lib/optionsFactory.js deleted file mode 100644 index 7284085..0000000 --- a/extensions/46/vertical-workspaces/lib/optionsFactory.js +++ /dev/null @@ -1,496 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * optionsFactory.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - */ - -'use strict'; - -import Adw from 'gi://Adw'; -import Gio from 'gi://Gio'; -import GObject from 'gi://GObject'; -import Gtk from 'gi://Gtk'; - -let Me; - -// gettext -let _; - -export function init(me) { - Me = me; - _ = Me.gettext; -} - -export const ItemFactory = class ItemFactory { - constructor() { - this._settings = Me.Opt._gsettings; - } - - getRowWidget(text, caption, widget, variable, options = [], dependsOn) { - let item = []; - let label; - if (widget) { - label = new Gtk.Box({ - orientation: Gtk.Orientation.VERTICAL, - spacing: 4, - halign: Gtk.Align.START, - valign: Gtk.Align.CENTER, - }); - const option = new Gtk.Label({ - halign: Gtk.Align.START, - }); - option.set_text(text); - label.append(option); - - if (caption) { - const captionLabel = new Gtk.Label({ - halign: Gtk.Align.START, - wrap: true, - /* width_chars: 80, */ - xalign: 0, - }); - const context = captionLabel.get_style_context(); - context.add_class('dim-label'); - context.add_class('caption'); - captionLabel.set_text(caption); - label.append(captionLabel); - } - label._title = text; - } else { - label = text; - } - item.push(label); - item.push(widget); - - let key; - - if (variable && Me.Opt.options[variable]) { - const opt = Me.Opt.options[variable]; - key = opt[1]; - } - - if (widget) { - if (widget._isSwitch) - this._connectSwitch(widget, key, variable); - else if (widget._isSpinButton || widget._isScale) - this._connectSpinButton(widget, key, variable); - else if (widget._isComboBox) - this._connectComboBox(widget, key, variable, options); - else if (widget._isDropDown) - this._connectDropDown(widget, key, variable, options); - - if (dependsOn) { - const dKey = Me.Opt.options[dependsOn][1]; - this._settings.bind(dKey, widget, 'sensitive', Gio.SettingsBindFlags.GET); - } - } - - return item; - } - - _connectSwitch(widget, key /* , variable */) { - this._settings.bind(key, widget, 'active', Gio.SettingsBindFlags.DEFAULT); - } - - _connectSpinButton(widget, key /* , variable */) { - this._settings.bind(key, widget.adjustment, 'value', Gio.SettingsBindFlags.DEFAULT); - } - - _connectComboBox(widget, key, variable, options) { - let model = widget.get_model(); - widget._comboMap = {}; - const currentValue = Me.Opt.get(variable); - for (const [label, value] of options) { - let iter; - model.set(iter = model.append(), [0, 1], [label, value]); - if (value === currentValue) - widget.set_active_iter(iter); - - widget._comboMap[value] = iter; - } - Me.Opt.connect(`changed::${key}`, () => { - widget.set_active_iter(widget._comboMap[Me.Opt.get(variable, true)]); - }); - widget.connect('changed', () => { - const [success, iter] = widget.get_active_iter(); - - if (!success) - return; - - Me.Opt.set(variable, model.get_value(iter, 1)); - }); - } - - _connectDropDown(widget, key, variable, options) { - const model = widget.get_model(); - const currentValue = Me.Opt.get(variable); - for (let i = 0; i < options.length; i++) { - const text = options[i][0]; - const id = options[i][1]; - model.append(new DropDownItem({ text, id })); - if (id === currentValue) - widget.set_selected(i); - } - - const factory = new Gtk.SignalListItemFactory(); - factory.connect('setup', (fact, listItem) => { - const label = new Gtk.Label({ xalign: 0 }); - listItem.set_child(label); - }); - factory.connect('bind', (fact, listItem) => { - const label = listItem.get_child(); - const item = listItem.get_item(); - label.set_text(item.text); - }); - - widget.connect('notify::selected-item', dropDown => { - const item = dropDown.get_selected_item(); - Me.Opt.set(variable, item.id); - }); - - Me.Opt.connect(`changed::${key}`, () => { - const newId = Me.Opt.get(variable, true); - for (let i = 0; i < options.length; i++) { - const id = options[i][1]; - if (id === newId) - widget.set_selected(i); - } - }); - - widget.set_factory(factory); - } - - newSwitch() { - let sw = new Gtk.Switch({ - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - }); - sw._isSwitch = true; - return sw; - } - - newSpinButton(adjustment) { - let spinButton = new Gtk.SpinButton({ - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - vexpand: false, - xalign: 0.5, - }); - spinButton.set_adjustment(adjustment); - spinButton._isSpinButton = true; - return spinButton; - } - - newComboBox() { - const model = new Gtk.ListStore(); - model.set_column_types([GObject.TYPE_STRING, GObject.TYPE_INT]); - const comboBox = new Gtk.ComboBox({ - model, - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - }); - const renderer = new Gtk.CellRendererText(); - comboBox.pack_start(renderer, true); - comboBox.add_attribute(renderer, 'text', 0); - comboBox._isComboBox = true; - return comboBox; - } - - newDropDown() { - const dropDown = new Gtk.DropDown({ - model: new Gio.ListStore({ - item_type: DropDownItem, - }), - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - }); - dropDown._isDropDown = true; - return dropDown; - } - - newScale(adjustment) { - const scale = new Gtk.Scale({ - orientation: Gtk.Orientation.HORIZONTAL, - draw_value: true, - has_origin: false, - value_pos: Gtk.PositionType.LEFT, - digits: 0, - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - vexpand: false, - }); - scale.set_size_request(300, -1); - scale.set_adjustment(adjustment); - scale._isScale = true; - return scale; - } - - newLabel(text = '') { - const label = new Gtk.Label({ - label: text, - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - }); - label._activatable = false; - return label; - } - - newLinkButton(uri) { - const linkBtn = new Gtk.LinkButton({ - uri, - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - icon_name: 'emblem-symbolic-link', - }); - return linkBtn; - } - - newButton() { - const btn = new Gtk.Button({ - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - }); - - btn._activatable = true; - return btn; - } - - newPresetButton(opt, profileIndex) { - const load = opt.loadProfile.bind(opt); - const save = opt.storeProfile.bind(opt); - const reset = opt.resetProfile.bind(opt); - - const box = new Gtk.Box({ - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - spacing: 8, - }); - box.is_profile_box = true; - - const entry = new Gtk.Entry({ - width_chars: 40, - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - xalign: 0, - }); - entry.set_text(opt.get(`profileName${profileIndex}`)); - entry.set_icon_from_icon_name(Gtk.EntryIconPosition.SECONDARY, 'edit-clear-symbolic'); - entry.set_icon_activatable(Gtk.EntryIconPosition.SECONDARY, true); - - const resetProfile = this.newButton(); - resetProfile.set({ - tooltip_text: _('Reset profile to defaults'), - icon_name: 'document-revert-symbolic', - hexpand: false, - css_classes: ['destructive-action'], - }); - - function setName() { - const ProfileNames = [ - _('GNOME 3 Layout (Vertical WS)'), - _('GNOME 4x Layout, Bottom Hot Edge (Horizontal WS)'), - _('Top Left Hot Corner Centric (Vertical WS)'), - _('Dock-Like Overview, Bottom Hot Edge (Horizontal WS)'), - ]; - - let name = opt.get(`profileName${profileIndex}`, true); - if (!name) - name = ProfileNames[profileIndex - 1]; - entry.set_text(name); - } - - setName(); - - entry.connect('icon-press', e => e.set_text('')); - entry.connect('changed', e => opt.set(`profileName${profileIndex}`, e.get_text())); - - resetProfile.connect('clicked', () => { - reset(profileIndex); - setName(); - }); - resetProfile._activatable = false; - - const loadProfile = this.newButton(); - loadProfile.set({ - tooltip_text: _('Load profile'), - icon_name: 'view-refresh-symbolic', - hexpand: false, - }); - loadProfile.connect('clicked', () => load(profileIndex)); - loadProfile._activatable = false; - - const saveProfile = this.newButton(); - saveProfile.set({ - tooltip_text: _('Save current settings into this profile'), - icon_name: 'document-save-symbolic', - hexpand: false, - }); - saveProfile.connect('clicked', () => save(profileIndex)); - saveProfile._activatable = false; - - box.append(resetProfile); - box.append(entry); - box.append(saveProfile); - box.append(loadProfile); - return box; - } - - newResetButton(callback) { - const btn = this.newButton(); - btn.set({ - css_classes: ['destructive-action'], - icon_name: 'edit-delete-symbolic', - }); - - btn.connect('clicked', callback); - btn._activatable = false; - return btn; - } - - newOptionsResetButton() { - const btn = new Gtk.Button({ - halign: Gtk.Align.END, - valign: Gtk.Align.CENTER, - hexpand: true, - css_classes: ['destructive-action'], - icon_name: 'document-revert-symbolic', - }); - - btn.connect('clicked', () => { - const settings = this._settings; - settings.list_keys().forEach( - key => settings.reset(key) - ); - }); - btn._activatable = false; - return btn; - } -}; - -export const AdwPrefs = class { - constructor(gOptions) { - Me.Opt = gOptions; - } - - getFilledWindow(window, pages) { - for (let page of pages) { - const title = page.title; - const iconName = page.iconName; - const optionList = page.optionList; - - window.add( - this._getAdwPage(optionList, { - title, - icon_name: iconName, - }) - ); - } - - window.set_search_enabled(true); - - return window; - } - - _getAdwPage(optionList, pageProperties = {}) { - // pageProperties.width_request = 740; - const page = new Adw.PreferencesPage(pageProperties); - let group; - for (let item of optionList) { - // label can be plain text for Section Title - // or GtkBox for Option - const option = item[0]; - const widget = item[1]; - if (!widget) { - if (group) - page.add(group); - - group = new Adw.PreferencesGroup({ - title: option, - hexpand: true, - width_request: 700, - }); - continue; - } - - const row = new Adw.ActionRow({ - title: option._title, - }); - - const grid = new Gtk.Grid({ - column_homogeneous: false, - column_spacing: 20, - margin_start: 8, - margin_end: 8, - margin_top: 8, - margin_bottom: 8, - hexpand: true, - }); - /* for (let i of item) { - box.append(i);*/ - grid.attach(option, 0, 0, 1, 1); - if (widget) - grid.attach(widget, 1, 0, 1, 1); - - row.set_child(grid); - if (widget._activatable === false) - row.activatable = false; - else - row.activatable_widget = widget; - - group.add(row); - } - page.add(group); - return page; - } -}; - -const DropDownItem = GObject.registerClass({ - // Registered name should be unique - GTypeName: `DropDownItem${Math.floor(Math.random() * 1000)}`, - Properties: { - 'text': GObject.ParamSpec.string( - 'text', - 'Text', - 'DropDown item text', - GObject.ParamFlags.READWRITE, - '' - ), - 'id': GObject.ParamSpec.int( - 'id', - 'Id', - 'Item id stored in settings', - GObject.ParamFlags.READWRITE, - // min, max, default - -2147483648, 2147483647, 0 - ), - }, -}, class DropDownItem extends GObject.Object { - get text() { - return this._text; - } - - set text(text) { - this._text = text; - } - - get id() { - return this._id; - } - - set id(id) { - this._id = id; - } -}); diff --git a/extensions/46/vertical-workspaces/lib/osdWindow.js b/extensions/46/vertical-workspaces/lib/osdWindow.js deleted file mode 100644 index a06a331..0000000 --- a/extensions/46/vertical-workspaces/lib/osdWindow.js +++ /dev/null @@ -1,118 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * osdWindow.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as OsdWindow from 'resource:///org/gnome/shell/ui/osdWindow.js'; - -let Me; -let opt; - -let OsdPositions; - -export const OsdWindowModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - - OsdPositions = { - 1: { - x_align: Clutter.ActorAlign.START, - y_align: Clutter.ActorAlign.START, - }, - 2: { - x_align: Clutter.ActorAlign.CENTER, - y_align: Clutter.ActorAlign.START, - }, - 3: { - x_align: Clutter.ActorAlign.END, - y_align: Clutter.ActorAlign.START, - }, - 4: { - x_align: Clutter.ActorAlign.CENTER, - y_align: Clutter.ActorAlign.CENTER, - }, - 5: { - x_align: Clutter.ActorAlign.START, - y_align: Clutter.ActorAlign.END, - }, - 6: { - x_align: Clutter.ActorAlign.CENTER, - y_align: Clutter.ActorAlign.END, - }, - 7: { - x_align: Clutter.ActorAlign.END, - y_align: Clutter.ActorAlign.END, - }, - }; - } - - cleanGlobals() { - Me = null; - opt = null; - OsdPositions = null; - } - - update(reset) { - this.moduleEnabled = opt.get('osdWindowModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' OsdWindowModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('osdWindow', OsdWindow.OsdWindow.prototype, OsdWindowCommon); - console.debug(' OsdWindowModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - this._updateExistingOsdWindows(6); - - console.debug(' WorkspaceSwitcherPopupModule - Disabled'); - } - - _updateExistingOsdWindows(position) { - position = position ? position : opt.OSD_POSITION; - Main.osdWindowManager._osdWindows.forEach(osd => { - osd.set(OsdPositions[position]); - }); - } -}; - -const OsdWindowCommon = { - after_show() { - if (!opt.OSD_POSITION) - this.opacity = 0; - this.set(OsdPositions[opt.OSD_POSITION]); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/overlayKey.js b/extensions/46/vertical-workspaces/lib/overlayKey.js deleted file mode 100644 index 5ffd973..0000000 --- a/extensions/46/vertical-workspaces/lib/overlayKey.js +++ /dev/null @@ -1,170 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * overlayKey.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import GLib from 'gi://GLib'; -import St from 'gi://St'; -import Meta from 'gi://Meta'; -import GObject from 'gi://GObject'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Overview from 'resource:///org/gnome/shell/ui/overview.js'; - -let Me; -let opt; - -export const OverlayKeyModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._originalOverlayKeyHandlerId = 0; - this._overlayKeyHandlerId = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('overlayKeyModule'); - const conflict = false; - // Avoid modifying the overlay key if its configuration is consistent with the GNOME default - const defaultConfig = opt.OVERVIEW_MODE === 0 && opt.OVERLAY_KEY_PRIMARY === 2 && opt.OVERLAY_KEY_SECONDARY === 1; - - reset = reset || !this.moduleEnabled || conflict || defaultConfig; - - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' OverlayKeyModule - Keeping untouched'); - } - - _activateModule() { - if (!this._originalOverlayKeyHandlerId) { - this._originalOverlayKeyHandlerId = GObject.signal_handler_find(global.display, { signalId: 'overlay-key' }); - if (this._originalOverlayKeyHandlerId !== null) { - global.display.block_signal_handler(this._originalOverlayKeyHandlerId); - this._connectOverlayKey(); - } - } - console.debug(' OverlayKeyModule - Activated'); - } - - _disableModule() { - this._restoreOverlayKeyHandler(); - - console.debug(' OverlayKeyModule - Disabled'); - } - - _restoreOverlayKeyHandler() { - // Disconnect modified overlay key handler - if (this._overlayKeyHandlerId) { - global.display.disconnect(this._overlayKeyHandlerId); - this._overlayKeyHandlerId = 0; - } - - // Unblock original overlay key handler - if (this._originalOverlayKeyHandlerId) { - global.display.unblock_signal_handler(this._originalOverlayKeyHandlerId); - this._originalOverlayKeyHandlerId = 0; - } - } - - _connectOverlayKey() { - if (this._overlayKeyHandlerId) - return; - - this._overlayKeyHandlerId = global.display.connect('overlay-key', this._onOverlayKeyPressed.bind(Main.overview._overview.controls)); - } - - _onOverlayKeyPressed() { - if (this._a11ySettings.get_boolean('stickykeys-enable')) - return; - - const { initialState, finalState, transitioning } = - this._stateAdjustment.getStateTransitionParams(); - - const time = GLib.get_monotonic_time() / 1000; - const timeDiff = time - this._lastOverlayKeyTime; - this._lastOverlayKeyTime = time; - - const shouldShift = St.Settings.get().enable_animations - ? transitioning && finalState > initialState - : Main.overview.visible && timeDiff < Overview.ANIMATION_TIME; - - const mode = opt.OVERLAY_KEY_SECONDARY; - if (shouldShift) { - Me.Util.activateSearchProvider(''); - if (mode === 1) { - this._shiftState(Meta.MotionDirection.UP); - } else if (mode === 2) { - Me.Util.activateSearchProvider(Me.WSP_PREFIX); - } else if (mode === 3) { - // Changing the overview mode automatically changes the overview transition - opt.OVERVIEW_MODE = 0; - opt.OVERVIEW_MODE2 = false; - opt.WORKSPACE_MODE = 1; - } - } else { - if (Main.overview._shown) { - Main.overview.hide(); - return; - } - switch (opt.OVERLAY_KEY_PRIMARY) { - case 0: // Disabled - return; - case 1: // Follow global overview mode - Main.overview.resetOverviewMode(); - break; - case 2: // Default overview - opt.OVERVIEW_MODE = 0; - opt.OVERVIEW_MODE2 = false; - opt.WORKSPACE_MODE = 1; - break; - case 3: // App grid - if (Main.overview._shown) - Main.overview.hide(); - else - Main.overview.show(2); - return; - case 4: // Static WS preview - opt.OVERVIEW_MODE = 1; - opt.OVERVIEW_MODE2 = false; - if (!Main.overview._shown) - opt.WORKSPACE_MODE = 0; - break; - case 5: // Static WS - opt.OVERVIEW_MODE = 2; - opt.OVERVIEW_MODE2 = true; - opt.WORKSPACE_MODE = 0; - break; - case 6: // Window Search - opt.OVERVIEW_MODE = 2; - opt.OVERVIEW_MODE2 = true; - if (!Main.overview._shown) - opt.WORKSPACE_MODE = 0; - break; - } - const customOverviewMode = !Main.overview._shown; - Main.overview.toggle(customOverviewMode); - if (opt.OVERLAY_KEY_PRIMARY === 6) - Me.Util.activateSearchProvider(Me.WSP_PREFIX); - } - } -}; diff --git a/extensions/46/vertical-workspaces/lib/overview.js b/extensions/46/vertical-workspaces/lib/overview.js deleted file mode 100644 index 30cc5db..0000000 --- a/extensions/46/vertical-workspaces/lib/overview.js +++ /dev/null @@ -1,162 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * overview.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Overview from 'resource:///org/gnome/shell/ui/overview.js'; -import * as OverviewControls from 'resource:///org/gnome/shell/ui/overviewControls.js'; - -let Me; -let opt; - -export const OverviewModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = true; - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' OverviewModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('Overview', Overview.Overview.prototype, OverviewCommon); - console.debug(' OverviewModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - console.debug(' OverviewModule - Disabled'); - } -}; - -const OverviewCommon = { - show(state = OverviewControls.ControlsState.WINDOW_PICKER, customOverviewMode) { - if (!customOverviewMode) - this.resetOverviewMode(); - - if (state === OverviewControls.ControlsState.HIDDEN) - throw new Error('Invalid state, use hide() to hide'); - - if (this.isDummy) - return; - if (this._shown) - return; - this._shown = true; - - if (!this._syncGrab()) - return; - - Main.layoutManager.showOverview(); - this._animateVisible(state); - }, - - toggle(customOverviewMode) { - if (this.isDummy) - return; - - if (this._visible) - this.hide(); - else - this.show(OverviewControls.ControlsState.WINDOW_PICKER, customOverviewMode); - }, - - resetOverviewMode() { - // reset Overview Mode do default - opt.OVERVIEW_MODE = opt.get('overviewMode'); - opt.OVERVIEW_MODE2 = opt.OVERVIEW_MODE === 2; - opt.WORKSPACE_MODE = opt.OVERVIEW_MODE > 0 ? 0 : 1; - }, - - _showDone() { - this._animationInProgress = false; - this._coverPane.hide(); - - if (this._shownState !== 'SHOWN') - this._changeShownState('SHOWN'); - - // Handle any calls to hide* while we were showing - if (!this._shown) - this._animateNotVisible(); - - // if user activates overview during startup animation, transition needs to be shifted to the state 2 here - const controls = this._overview._controls; - if (controls._searchController._searchActive && controls._stateAdjustment.value === 1) { - if (opt.SEARCH_VIEW_ANIMATION) - controls._onSearchChanged(); - else if (!opt.OVERVIEW_MODE2) - controls._stateAdjustment.value = 2; - } - - this._syncGrab(); - }, - - // Workaround - should probably be fixed elsewhere in the upstream code - // If a new window is opened from the overview - // and is realized before the overview animation is complete, - // the new window will not get focus - after__hideDone() { - if (!opt.FIX_NEW_WINDOW_FOCUS) - return; - - const workspace = global.workspace_manager.get_active_workspace(); - const recentDesktopWin = global.display.get_tab_list(1, workspace)[0]; - let recentNormalWin = null; - const tabList = global.display.get_tab_list(0, workspace); - - for (let i = 0; i < tabList.length; i++) { - if (tabList[i].minimized === false) { - recentNormalWin = tabList[i]; - break; - } - } - - let recentWin = recentNormalWin; - if (recentNormalWin && recentDesktopWin) { - recentWin = recentNormalWin.get_user_time() > recentDesktopWin.get_user_time() - ? recentNormalWin - : recentDesktopWin; - } - - const focusedWin = global.display.focus_window; - - if (recentWin && focusedWin !== recentWin) - recentWin.activate(global.get_current_time()); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/overviewControls.js b/extensions/46/vertical-workspaces/lib/overviewControls.js deleted file mode 100644 index c5a74f1..0000000 --- a/extensions/46/vertical-workspaces/lib/overviewControls.js +++ /dev/null @@ -1,1633 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * overviewControls.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import GLib from 'gi://GLib'; -import GObject from 'gi://GObject'; -import St from 'gi://St'; -import Shell from 'gi://Shell'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Overview from 'resource:///org/gnome/shell/ui/overview.js'; -import * as OverviewControls from 'resource:///org/gnome/shell/ui/overviewControls.js'; -import * as WorkspacesView from 'resource:///org/gnome/shell/ui/workspacesView.js'; -import * as Background from 'resource:///org/gnome/shell/ui/background.js'; -import * as Util from 'resource:///org/gnome/shell/misc/util.js'; - -let Me; -let opt; -// gettext -let _; - -const ControlsState = OverviewControls.ControlsState; -const FitMode = WorkspacesView.FitMode; - -const STARTUP_ANIMATION_TIME = 500; -const ANIMATION_TIME = Overview.ANIMATION_TIME; -const DASH_MAX_SIZE_RATIO = 0.35; - -let _timeouts; - -export const OverviewControlsModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - _ = Me.gettext; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - _ = null; - } - - update(reset) { - this._removeTimeouts(); - this.moduleEnabled = true; - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' OverviewControlsModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - _timeouts = {}; - - this._replaceOnSearchChanged(); - - this._overrides.addOverride('ControlsManager', OverviewControls.ControlsManager.prototype, ControlsManagerCommon); - this._overrides.addOverride('ControlsManagerLayoutCommon', Main.overview._overview.controls.layoutManager, ControlsManagerLayoutCommon); - if (opt.ORIENTATION === Clutter.Orientation.VERTICAL) - this._overrides.addOverride('ControlsManagerLayout', Main.overview._overview.controls.layoutManager, ControlsManagerLayoutVertical); - else - this._overrides.addOverride('ControlsManagerLayout', Main.overview._overview.controls.layoutManager, ControlsManagerLayoutHorizontal); - - // Allow user to close the overview by clicking on an empty space on the primary monitor's overview - // Secondary monitors are handled in workspacesView - this._addClickToCloseOverview(); - - // Update custom workAreaBox - Main.overview._overview.controls.layoutManager._updateWorkAreaBox(); - - console.debug(' OverviewControlsModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - const reset = true; - this._replaceOnSearchChanged(reset); - Main.overview._overview._controls._appDisplay.opacity = 255; - this._addClickToCloseOverview(reset); - - console.debug(' OverviewControlsModule - Disabled'); - } - - _removeTimeouts() { - if (_timeouts) { - Object.values(_timeouts).forEach(t => { - if (t) - GLib.source_remove(t); - }); - _timeouts = null; - } - } - - _replaceOnSearchChanged(reset) { - const searchController = Main.overview.searchController; - if (reset) { - if (this._searchControllerSigId) { - searchController.disconnect(this._searchControllerSigId); - this._searchControllerSigId = 0; - } - if (this._originalSearchControllerSigId) { - searchController.unblock_signal_handler(this._originalSearchControllerSigId); - this._originalSearchControllerSigId = 0; - } - searchController._searchResults.translation_x = 0; - searchController._searchResults.translation_y = 0; - Main.overview.searchEntry.visible = true; - Main.overview.searchEntry.opacity = 255; - } else { - // reconnect signal to use custom function (callbacks cannot be overridden in class prototype, they are already in memory as a copy for the given callback) - if (!this._originalSearchControllerSigId) { - this._originalSearchControllerSigId = GObject.signal_handler_find(searchController, { signalId: 'notify', detail: 'search-active' }); - if (this._originalSearchControllerSigId) - searchController.block_signal_handler(this._originalSearchControllerSigId); - } - - if (!this._searchControllerSigId) - this._searchControllerSigId = searchController.connect('notify::search-active', () => Main.overview._overview.controls._onSearchChanged()); - } - } - - _addClickToCloseOverview(reset) { - const overview = Main.overview._overview; - - overview.reactive = false; - if (this._clickEmptyConId) { - overview.disconnect(this._clickEmptyConId); - this._clickEmptyConId = 0; - } - - if (reset || !opt.CLICK_EMPTY_CLOSE) - return; - - overview.reactive = true; - this._clickEmptyConId = overview.connect('button-release-event', (actor, event) => { - const button = event.get_button(); - const overviewState = overview.controls._stateAdjustment.value; - const buttonPrimary = button === Clutter.BUTTON_PRIMARY; - const buttonSecondary = button === Clutter.BUTTON_SECONDARY; - const buttonAny = buttonPrimary || buttonSecondary; - - if ((overviewState === 1 && buttonAny) || (overviewState === 2 && buttonSecondary)) - Main.overview.hide(); - }); - } -}; - -const ControlsManagerCommon = { - // this function is used as a callback by a signal handler, needs to be reconnected after modification as the original callback uses a copy of the original function - /* _update: function() { - ... - }*/ - - prepareToEnterOverview() { - this._searchController.prepareToEnterOverview(); - this._workspacesDisplay.prepareToEnterOverview(); - // Workaround for thumbnailsBox not re-scaling after switching workspace outside of overview using a trackpad - this._thumbnailsBox._updateIndicator(); - - Main.overview._overview.controls.opacity = 255; - - // Ensure that overview backgrounds are ready when needed - if (!this._bgManagers && (opt.SHOW_BG_IN_OVERVIEW || !opt.SHOW_WS_PREVIEW_BG)) - this._setBackground(); - else if (this._bgManagers && !(opt.SHOW_BG_IN_OVERVIEW || !opt.SHOW_WS_PREVIEW_BG)) - this._setBackground(true); - - // store pointer X coordinate for OVERVIEW_MODE 1 - to prevent immediate switch to WORKSPACE_MODE 1 if the mouse pointer is steady - opt.showingPointerX = global.get_pointer()[0]; - }, - - // this function has duplicate in WorkspaceView so we use one function for both to avoid issues with syncing them - _getFitModeForState(state) { - return _getFitModeForState(state); - }, - - _updateThumbnailsBox() { - const { shouldShow } = this._thumbnailsBox; - const thumbnailsBoxVisible = shouldShow; - this._thumbnailsBox.visible = thumbnailsBoxVisible; - - // this call should be directly in _update(), but it's used as a callback function and it would require to reconnect the signal - this._updateOverview(); - }, - - // this function is pure addition to the original code and handles wsDisp transition to APP_GRID view - _updateOverview() { - this._workspacesDisplay.translation_x = 0; - this._workspacesDisplay.translation_y = 0; - this._workspacesDisplay.scale_x = 1; - this._workspacesDisplay.scale_y = 1; - const { initialState, finalState, progress, currentState } = this._stateAdjustment.getStateTransitionParams(); - - const paramsForState = s => { - let opacity; - switch (s) { - case ControlsState.HIDDEN: - case ControlsState.WINDOW_PICKER: - opacity = 255; - break; - case ControlsState.APP_GRID: - opacity = 0; - break; - default: - opacity = 255; - break; - } - return { opacity }; - }; - - let initialParams = paramsForState(initialState); - let finalParams = paramsForState(finalState); - - let opacity = Math.round(Util.lerp(initialParams.opacity, finalParams.opacity, progress)); - - let workspacesDisplayVisible = opacity !== 0; - - // improve transition from search results to desktop - if (finalState === 0 && this._searchController._searchResults.visible) - this._searchController.hide(); - - // reset Static Workspace window picker mode - if (currentState === 0 && opt.OVERVIEW_MODE && opt.WORKSPACE_MODE) - opt.WORKSPACE_MODE = 0; - - if (!opt.WS_ANIMATION || (!opt.SHOW_WS_TMB && opt.SHOW_WS_PREVIEW_BG)) { - this._workspacesDisplay.opacity = opacity; - } else if (!opt.SHOW_WS_TMB_BG && opt.SHOW_WS_PREVIEW_BG) { - // fade out ws wallpaper during transition to ws switcher if ws switcher background disabled - const workspaces = this._workspacesDisplay._workspacesViews[global.display.get_primary_monitor()]?._workspaces; - // Speed up the workspace background opacity transition - if (opt.WORKSPACE_MAX_SPACING < opt.WS_MAX_SPACING_OFF_SCREEN && workspaces) - // If workspacesDisplay max spacing is set so adjacent workspaces could be visible on the screen - workspaces.forEach(w => w._background.set_opacity(Math.max(0, opacity - (255 - opacity)))); - else if (workspaces) - // If adjacent workspaces should not be visible on the screen, set the opacity only for the visible one - workspaces[this._workspaceAdjustment.value]?._background.set_opacity(Math.max(0, opacity - (255 - opacity))); - } - - // if ws preview background is disabled, animate tmb box and dash - const tmbBox = this._thumbnailsBox; - const dash = this.dash; - const searchEntryBin = this._searchEntryBin; - // this dash transition collides with startup animation and freezes GS for good, needs to be delayed (first Main.overview 'hiding' event enables it) - const skipDash = Me.Util.dashNotDefault(); - - // OVERVIEW_MODE 2 should animate dash and wsTmbBox only if WORKSPACE_MODE === 0 (windows not spread) - const animateOverviewMode2 = opt.OVERVIEW_MODE2 && !(finalState === 1 && opt.WORKSPACE_MODE); - if (!Main.layoutManager._startingUp && ((!opt.SHOW_WS_PREVIEW_BG && !opt.OVERVIEW_MODE2) || animateOverviewMode2)) { - if (!tmbBox._translationOriginal || Math.abs(tmbBox._translationOriginal[0]) > 500) { // swipe gesture can call this calculation before tmbBox is realized, giving nonsense width - const [dashTranslationX, dashTranslationY, tmbTranslationX, tmbTranslationY, searchTranslationY] = this._getOverviewTranslations(dash, tmbBox, searchEntryBin); - tmbBox._translationOriginal = [tmbTranslationX, tmbTranslationY]; - dash._translationOriginal = [dashTranslationX, dashTranslationY]; - searchEntryBin._translationOriginal = searchTranslationY; - } - if (finalState === 0 || initialState === 0) { - const prg = Math.abs((finalState === 0 ? 0 : 1) - progress); - tmbBox.translation_x = Math.round(prg * tmbBox._translationOriginal[0]); - tmbBox.translation_y = Math.round(prg * tmbBox._translationOriginal[1]); - if (!skipDash) { - dash.translation_x = Math.round(prg * dash._translationOriginal[0]); - dash.translation_y = Math.round(prg * dash._translationOriginal[1]); - } - searchEntryBin.translation_y = Math.round(prg * searchEntryBin._translationOriginal); - } - if (progress === 1) { - tmbBox._translationOriginal = 0; - if (!skipDash) - dash._translationOriginal = 0; - - searchEntryBin._translationOriginal = 0; - } - } else if (!Main.layoutManager._startingUp && (tmbBox.translation_x || tmbBox.translation_y)) { - tmbBox.translation_x = 0; - tmbBox.translation_y = 0; - if (!skipDash) { - dash.translation_x = 0; - dash.translation_y = 0; - } - searchEntryBin.translation_y = 0; - } - - if (!Main.layoutManager._startingUp) { - if (initialState === ControlsState.HIDDEN && finalState === ControlsState.APP_GRID) - this._appDisplay.opacity = Math.round(progress * 255); - else - this._appDisplay.opacity = 255 - opacity; - } - - if (currentState === ControlsState.APP_GRID) { - // in app grid hide workspaces so they're not blocking app grid or ws thumbnails - this._workspacesDisplay.scale_x = 0; - } else { - this._workspacesDisplay.scale_x = 1; - } - if (opt.LEAVING_SEARCH && currentState <= ControlsState.WINDOW_PICKER) { - opt.LEAVING_SEARCH = false; - } - - this._workspacesDisplay.setPrimaryWorkspaceVisible(workspacesDisplayVisible); - - if (!this.dash._isAbove && progress > 0 && opt.OVERVIEW_MODE2) { - // set searchEntry above appDisplay - this.set_child_above_sibling(this._searchEntryBin, null); - // move dash above wsTmb for case that dash and wsTmb animate from the same side - if (!Me.Util.dashNotDefault()) - this.set_child_above_sibling(dash, null); - this.set_child_below_sibling(this._thumbnailsBox, null); - this.set_child_below_sibling(this._workspacesDisplay, null); - this.set_child_below_sibling(this._appDisplay, null); - } else if (!this.dash._isAbove && progress === 1 && finalState > ControlsState.HIDDEN) { - // set dash above workspace in the overview - this.set_child_above_sibling(this._thumbnailsBox, null); - this.set_child_above_sibling(this._searchEntryBin, null); - if (!Me.Util.dashNotDefault()) - this.set_child_above_sibling(this.dash, null); - if (Main.layoutManager.panelBox.get_parent() === Main.layoutManager.overviewGroup) - Main.layoutManager.overviewGroup.set_child_above_sibling(Main.layoutManager.panelBox, Main.overview._overview); - this.dash._isAbove = true; - } else if (this.dash._isAbove && progress < 1) { - // keep dash below for ws transition between the overview and hidden state - this.set_child_above_sibling(this._workspacesDisplay, null); - if (Main.layoutManager.panelBox.get_parent() === Main.layoutManager.overviewGroup) - Main.layoutManager.overviewGroup.set_child_below_sibling(Main.layoutManager.panelBox, Main.overview._overview); - this.dash._isAbove = false; - } - }, - - // fix for upstream bug - appGrid.visible after transition from APP_GRID to HIDDEN - _updateAppDisplayVisibility(stateTransitionParams = null) { - if (!stateTransitionParams) - stateTransitionParams = this._stateAdjustment.getStateTransitionParams(); - - const { currentState } = stateTransitionParams; - if (this.dash.showAppsButton.checked) - this._searchTransition = false; - - // if !APP_GRID_ANIMATION, appGrid needs to be hidden in WINDOW_PICKER mode (1) - // but needs to be visible for transition from HIDDEN (0) to APP_GRID (2) - this._appDisplay.visible = - currentState > ControlsState.HIDDEN && - !this._searchController.searchActive && - !(currentState === ControlsState.WINDOW_PICKER && !opt.APP_GRID_ANIMATION) && - !this._searchTransition; - }, - - _activateSearchAppGridMode() { - if (!this._origAppGridContent) { - this._origAppGridContent = { - usage: opt.APP_GRID_USAGE, - favorites: opt.APP_GRID_EXCLUDE_FAVORITES, - running: opt.APP_GRID_EXCLUDE_RUNNING, - incompletePages: this._appDisplay._grid.layoutManager.allowIncompletePages, - order: opt.APP_GRID_ORDER, - }; - opt.APP_GRID_ORDER = 3; - opt.APP_GRID_USAGE = true; - opt.APP_GRID_EXCLUDE_FAVORITES = false; - opt.APP_GRID_EXCLUDE_RUNNING = false; - this._appDisplay._grid.layoutManager.allowIncompletePages = false; - this._appDisplay._redisplay(); - } - }, - - _deactivateSearchAppGridMode() { - if (this._origAppGridContent) { - const icons = this._appDisplay._orderedItems; - icons.forEach(icon => { - icon.visible = true; - }); - - opt.APP_GRID_ORDER = this._origAppGridContent.order; - opt.APP_GRID_USAGE = this._origAppGridContent.usage; - opt.APP_GRID_EXCLUDE_FAVORITES = this._origAppGridContent.favorites; - opt.APP_GRID_EXCLUDE_RUNNING = this._origAppGridContent.running; - this._appDisplay._grid.layoutManager.allowIncompletePages = this._origAppGridContent.incompletePages; - this._origAppGridContent = null; - this._appDisplay._redisplay(); - } - }, - - _onSearchChanged() { - // something is somewhere setting the opacity to 0 if V-Shell is rebased while in overview / search - this._searchController.opacity = 255; - - const { finalState, currentState } = this._stateAdjustment.getStateTransitionParams(); - - const { searchActive } = this._searchController; - const SIDE_CONTROLS_ANIMATION_TIME = 250; // OverviewControls.SIDE_CONTROLS_ANIMATION_TIME = Overview.ANIMATION_TIME = 250 - - const entry = this._searchEntry; - if (opt.SHOW_SEARCH_ENTRY) { - entry.visible = true; - entry.opacity = 255; - } else if (!(searchActive && entry.visible)) { - entry.visible = true; - entry.opacity = searchActive ? 0 : 255; - // show search entry only if the user starts typing, and hide it when leaving the search mode - entry.ease({ - opacity: searchActive ? 255 : 0, - duration: SIDE_CONTROLS_ANIMATION_TIME / 2, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - entry.visible = searchActive; - }, - }); - } - - // if user start typing or activated search provider during overview animation, this switcher will be called again after animation ends - if (opt.SEARCH_VIEW_ANIMATION && Main.overview._animationInProgress && finalState !== ControlsState.HIDDEN) - return; - - if (!searchActive) { - if (!this.dash.showAppsButton.checked) - opt.LEAVING_SEARCH = true; - - if (this._origAppGridContent) - this._deactivateSearchAppGridMode(); - - this._workspacesDisplay.reactive = true; - this._workspacesDisplay.setPrimaryWorkspaceVisible(true); - } else { - if (opt.SEARCH_APP_GRID_MODE && this.dash.showAppsButton.checked) { - this._activateSearchAppGridMode(); - return; - } - - if (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE) - this._searchController._searchResults._statusText.add_style_class_name('search-statustext-om2'); - else - this._searchController._searchResults._statusText.remove_style_class_name('search-statustext-om2'); - this._searchController.show(); - entry.visible = true; - entry.opacity = 255; - opt.LEAVING_SEARCH = false; - } - - if (opt.SHOW_BG_IN_OVERVIEW && this._bgManagers) - this._updateBackground(this._bgManagers[0]); - this._searchTransition = true; - - this._searchController._searchResults.translation_x = 0; - this._searchController._searchResults.translation_y = 0; - this._searchController.visible = true; - - if (opt.SEARCH_VIEW_ANIMATION && ![4, 8].includes(opt.WS_TMB_POSITION)) { - this._updateAppDisplayVisibility(); - this._searchController._searchResults._statusBin.opacity = 1; - - let translationX = 0; - let translationY = 0; - const geometry = global.display.get_monitor_geometry(global.display.get_primary_monitor()); - - switch (opt.SEARCH_VIEW_ANIMATION) { - case 1: - // make it longer to cover the delay before results appears - translationX = geometry.width; - translationY = 0; - break; - case 2: - translationX = -geometry.width; - translationY = 0; - break; - case 3: - translationX = 0; - translationY = geometry.height; - break; - case 5: - translationX = 0; - translationY = -geometry.height; - break; - } - - if (searchActive) { - this._searchController._searchResults.translation_x = translationX; - this._searchController._searchResults.translation_y = translationY; - } else { - this._searchController._searchResults.translation_x = 0; - this._searchController._searchResults.translation_y = 0; - } - - this._searchController._searchResults.ease({ - delay: 150, // wait for results - opacity: searchActive ? 255 : 0, - translation_x: searchActive ? 0 : translationX, - translation_y: searchActive ? 0 : translationY, - duration: SIDE_CONTROLS_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._searchController.visible = searchActive; - this._searchTransition = false; - this._searchController._searchResults._statusBin.opacity = 255; - }, - }); - - this._workspacesDisplay.opacity = 255; - } else { - this._appDisplay.ease({ - opacity: searchActive || currentState < 2 ? 0 : 255, - duration: SIDE_CONTROLS_ANIMATION_TIME / 2, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._updateAppDisplayVisibility(); - }, - }); - - this._workspacesDisplay.setPrimaryWorkspaceVisible(true); - - this._searchController._searchResults.ease({ - opacity: searchActive ? 255 : 0, - duration: searchActive ? SIDE_CONTROLS_ANIMATION_TIME / 2 : 0, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => (this._searchController.visible = searchActive), - }); - } - - // reuse already tuned overview transition, just replace APP_GRID with the search view - if (!(opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE) && !Main.overview._animationInProgress && finalState !== ControlsState.HIDDEN && !this.dash.showAppsButton.checked) { - this._searchController._searchResults._content.remove_style_class_name('search-section-content-bg-om2'); - this._searchEntry.remove_style_class_name('search-entry-om2'); - const duration = opt.SEARCH_VIEW_ANIMATION ? 140 : 0; - this._stateAdjustment.ease(searchActive ? ControlsState.APP_GRID : ControlsState.WINDOW_PICKER, { - // shorter animation time when entering search view can avoid stuttering in transition - // collecting search results take some time and the problematic part is the realization of the object on the screen - // if the ws animation ends before this event, the whole transition is smoother - // removing the ws transition (duration: 0) seems like the best solution here - duration: searchActive ? duration : SIDE_CONTROLS_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._workspacesDisplay.setPrimaryWorkspaceVisible(!searchActive); - // Set the delay before processing a new search entry to 150 on deactivation, so search providers can't make make the workspace animation stuttering - // set it back to 0 after in-animation, so the search can be snappy - opt.SEARCH_DELAY = searchActive || !opt.SEARCH_VIEW_ANIMATION ? 0 : 150; - }, - }); - } else if (opt.OVERVIEW_MODE2 && !(opt.WORKSPACE_MODE || this.dash.showAppsButton.checked)) { - // add background to search results and make searchEntry border thicker for better visibility - this._searchController._searchResults._content.add_style_class_name('search-section-content-bg-om2'); - this._searchEntry.add_style_class_name('search-entry-om2'); - } else { - this._searchController._searchResults._content.remove_style_class_name('search-section-content-bg-om2'); - this._searchEntry.remove_style_class_name('search-entry-om2'); - } - }, - - async runStartupAnimation(callback) { - this._ignoreShowAppsButtonToggle = true; - - this.prepareToEnterOverview(); - - this._stateAdjustment.value = ControlsState.HIDDEN; - this._stateAdjustment.ease(ControlsState.WINDOW_PICKER, { - duration: ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - - this.dash.showAppsButton.checked = false; - this._ignoreShowAppsButtonToggle = false; - - // Set the opacity here to avoid a 1-frame flicker - this.opacity = 1; - this._appDisplay.opacity = 1; - - // We can't run the animation before the first allocation happens - await this.layout_manager.ensureAllocation(); - - this._setBackground(); - Me.Modules.panelModule.update(); - Main.panel.opacity = 255; - - // Opacity - this.ease({ - opacity: opt.STARTUP_STATE === 1 ? 0 : 255, - duration: STARTUP_ANIMATION_TIME, - mode: Clutter.AnimationMode.LINEAR, - }); - - const dash = this.dash; - const tmbBox = this._thumbnailsBox; - - // Set the opacity here to avoid a 1-frame flicker - dash.opacity = 0; - for (const view of this._workspacesDisplay._workspacesViews) { - if (view._monitorIndex !== global.display.get_primary_monitor()) - view._thumbnails.opacity = 0; - } - - const searchEntryBin = this._searchEntryBin; - const [dashTranslationX, dashTranslationY, tmbTranslationX, tmbTranslationY, searchTranslationY] = - this._getOverviewTranslations(dash, tmbBox, searchEntryBin); - - const onComplete = function () { - // running init callback again causes issues (multiple connections) - if (!Main.overview._startupInitComplete) - callback(); - - const appDisplayModule = Me.Modules.appDisplayModule; - if (!appDisplayModule.moduleEnabled) - this._finishStartupSequence(); - else - this._realizeAppDisplayAndFinishSequence(); - - Main.overview._startupInitComplete = true; - }.bind(this); - - if (dash.visible && !Me.Util.dashNotDefault()) { - dash.translation_x = dashTranslationX; - dash.translation_y = dashTranslationY; - dash.opacity = 255; - dash.ease({ - translation_x: 0, - translation_y: 0, - delay: STARTUP_ANIMATION_TIME / 2, - duration: STARTUP_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete, - }); - } else { - // set dash opacity to make it visible if user enable it later - dash.opacity = 255; - // if dash is hidden, substitute the ease timeout with GLib.timeout - _timeouts.startupAnim2 = GLib.timeout_add( - GLib.PRIORITY_DEFAULT, - // delay + animation time - STARTUP_ANIMATION_TIME * 2 * St.Settings.get().slow_down_factor, - () => { - onComplete(); - _timeouts.startupAnim2 = 0; - return GLib.SOURCE_REMOVE; - } - ); - } - - if (searchEntryBin.visible) { - searchEntryBin.translation_y = searchTranslationY; - searchEntryBin.ease({ - translation_y: 0, - delay: STARTUP_ANIMATION_TIME / 2, - duration: STARTUP_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - - if (tmbBox.visible) { - tmbBox.translation_x = tmbTranslationX; - tmbBox.translation_y = tmbTranslationY; - tmbBox.ease({ - translation_x: 0, - translation_y: 0, - delay: STARTUP_ANIMATION_TIME / 2, - duration: STARTUP_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - - // upstream bug - following animation will be cancelled, don't know where - // needs further investigation - const workspacesViews = this._workspacesDisplay._workspacesViews; - if (workspacesViews.length > 1) { - for (const view of workspacesViews) { - if (view._monitorIndex !== global.display.get_primary_monitor() && view._thumbnails.visible) { - const secTmbBox = view._thumbnails; - - if (opt.SEC_WS_TMB_LEFT) - secTmbBox.translation_x = -(secTmbBox.width + 12); // compensate for padding - else if (opt.SEC_WS_TMB_RIGHT) - secTmbBox.translation_x = secTmbBox.width + 12; - else if (opt.SEC_WS_TMB_TOP) - secTmbBox.translation_y = -(secTmbBox.height + 12); - else if (opt.SEC_WS_TMB_BOTTOM) - secTmbBox.translation_y = secTmbBox.height + 12; - - secTmbBox.opacity = 255; - - secTmbBox.ease({ - translation_y: 0, - delay: STARTUP_ANIMATION_TIME / 2, - duration: STARTUP_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - } - } - }, - - _realizeAppDisplayAndFinishSequence() { - const appDisplayModule = Me.Modules.appDisplayModule; - // realize app grid for smoother first animation - appDisplayModule._repopulateAppDisplay(false, this._finishStartupSequence.bind(this)); - }, - - _finishStartupSequence() { - if (!this._bgManagers) - this._setBackground(); - - _timeouts.finishStartup = GLib.idle_add( - GLib.PRIORITY_LOW, () => { - this._appDisplay.opacity = 255; - if (opt.STARTUP_STATE === 1) { - Main.overview.hide(); - } else if (opt.STARTUP_STATE === 2) { - Main.overview.show(2); // just because of DtD, because we skipped startup animation - this.dash.showAppsButton.checked = true; - } else if (!opt.STARTUP_STATE && Me.Util.dashNotDefault()) { - Main.overview.show(); - } - - _timeouts.finishStartup = 0; - return GLib.SOURCE_REMOVE; - } - ); - }, - - setInitialTranslations() { - const dash = this.dash; - const tmbBox = this._thumbnailsBox; - const searchEntryBin = this._searchEntryBin; - const [dashTranslationX, dashTranslationY, tmbTranslationX, tmbTranslationY, searchTranslationY] = - this._getOverviewTranslations(dash, tmbBox, searchEntryBin); - if (!Me.Util.dashNotDefault()) { - dash.translation_x = dashTranslationX; - dash.translation_y = dashTranslationY; - } - tmbBox.translation_x = tmbTranslationX; - tmbBox.translation_y = tmbTranslationY; - searchEntryBin.translation_y = searchTranslationY; - }, - - _getOverviewTranslations(dash, tmbBox, searchEntryBin) { - const animationsDisabled = !St.Settings.get().enable_animations || ((opt.SHOW_WS_PREVIEW_BG && !opt.OVERVIEW_MODE2) && !Main.layoutManager._startingUp); - if (animationsDisabled) - return [0, 0, 0, 0, 0]; - - let searchTranslationY = 0; - if (searchEntryBin.visible) { - const offset = (dash.visible && (!opt.DASH_VERTICAL ? dash.height + 12 : 0)) + - (opt.WS_TMB_TOP ? tmbBox.height + 12 : 0); - searchTranslationY = -searchEntryBin.height - offset - 30; - } - - let tmbTranslationX = 0; - let tmbTranslationY = 0; - let offset; - if (tmbBox.visible) { - const tmbWidth = tmbBox.width === Infinity ? 0 : tmbBox.width; - const tmbHeight = tmbBox.height === Infinity ? 0 : tmbBox.height; - switch (opt.WS_TMB_POSITION) { - case 3: // left - offset = 10 + (dash?.visible && opt.DASH_LEFT ? dash.width : 0); - tmbTranslationX = -tmbWidth - offset; - tmbTranslationY = 0; - break; - case 1: // right - offset = 10 + (dash?.visible && opt.DASH_RIGHT ? dash.width : 0); - tmbTranslationX = tmbWidth + offset; - tmbTranslationY = 0; - break; - case 0: // top - offset = 10 + (dash?.visible && opt.DASH_TOP ? dash.height : 0) + Main.panel.height; - tmbTranslationX = 0; - tmbTranslationY = -tmbHeight - offset; - break; - case 2: // bottom - offset = 10 + (dash?.visible && opt.DASH_BOTTOM ? dash.height : 0) + Main.panel.height; // just for case the panel is at bottom - tmbTranslationX = 0; - tmbTranslationY = tmbHeight + offset; - break; - } - } - - let dashTranslationX = 0; - let dashTranslationY = 0; - let position = opt.DASH_POSITION; - // if DtD replaced the original Dash, read its position - if (Me.Util.dashIsDashToDock()) - position = dash._position; - - if (dash?.visible) { - const dashWidth = dash.width === Infinity ? 0 : dash.width; - const dashHeight = dash.height === Infinity ? 0 : dash.height; - switch (position) { - case 0: // top - dashTranslationX = 0; - dashTranslationY = -dashHeight - dash.margin_bottom - Main.panel.height; - break; - case 1: // right - dashTranslationX = dashWidth; - dashTranslationY = 0; - break; - case 2: // bottom - dashTranslationX = 0; - dashTranslationY = dashHeight + dash.margin_bottom + Main.panel.height; - break; - case 3: // left - dashTranslationX = -dashWidth; - dashTranslationY = 0; - break; - } - } - - return [dashTranslationX, dashTranslationY, tmbTranslationX, tmbTranslationY, searchTranslationY]; - }, - - animateToOverview(state, callback) { - this._ignoreShowAppsButtonToggle = true; - this._searchTransition = false; - - this._stateAdjustment.value = ControlsState.HIDDEN; - - // building window thumbnails takes some time and with many windows on the workspace - // the time can be close to or longer than ANIMATION_TIME - // in which case the the animation is greatly delayed, stuttering, or even skipped - // for user it is more acceptable to watch delayed smooth animation, - // even if it takes little more time, than jumping frames - let delay = 0; - if (opt.DELAY_OVERVIEW_ANIMATION) - delay = global.display.get_tab_list(0, global.workspace_manager.get_active_workspace()).length * 3; - - this._stateAdjustment.ease(state, { - delay, - duration: 250, // Overview.ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onStopped: () => { - if (callback) - callback(); - }, - }); - - this.dash.showAppsButton.checked = - state === ControlsState.APP_GRID; - - this._ignoreShowAppsButtonToggle = false; - }, - - _setBackground(reset = false) { - if (this._bgManagers) { - this._bgManagers.forEach(bg => { - this._stateAdjustment.disconnect(bg._fadeSignal); - bg.destroy(); - }); - } - - // if (!SHOW_BG_IN_OVERVIEW && !SHOW_WS_PREVIEW_BG) the background is used for static transition from wallpaper to empty bg in the overview - if (reset || (!opt.SHOW_BG_IN_OVERVIEW && opt.SHOW_WS_PREVIEW_BG)) { - delete this._bgManagers; - return; - } - - this._bgManagers = []; - for (const monitor of Main.layoutManager.monitors) { - const bgManager = new Background.BackgroundManager({ - monitorIndex: monitor.index, - container: Main.layoutManager.overviewGroup, - vignette: true, - }); - - bgManager.backgroundActor.content.vignette_sharpness = 0; - bgManager.backgroundActor.content.brightness = 1; - - bgManager._fadeSignal = this._stateAdjustment.connect('notify::value', v => { - this._updateBackground(bgManager, v.value, v); - }); - - if (monitor.index === global.display.get_primary_monitor()) { - bgManager._primary = true; - this._bgManagers.unshift(bgManager); // primary monitor first - } else { - bgManager._primary = false; - this._bgManagers.push(bgManager); - } - } - }, - - _updateBackground(bgManager, stateValue = 2, stateAdjustment = null) { - // Just in case something destroys our background (like older versions of Blur My Shell) - if (this._bgManagers[0] && !Main.layoutManager.overviewGroup.get_children().includes(this._bgManagers[0].backgroundActor)) { - console.error(`[${Me.metadata.name}]`, 'Error: The overview background has been destroyed, possibly by another incompatible extension'); - // remove and disconnect our destroyed backgrounds to avoid further errors - this._setBackground(true); - return; - } - - const finalState = stateAdjustment?.getStateTransitionParams().finalState; - if (!opt.SHOW_BG_IN_OVERVIEW && !opt.SHOW_WS_PREVIEW_BG) { - // if no bg shown in the overview, fade out the wallpaper - if (bgManager.backgroundActor.get_effect('blur')) - bgManager.backgroundActor.remove_effect_by_name('blur'); - if (!(opt.OVERVIEW_MODE2 && opt.WORKSPACE_MODE && finalState === 1)) - bgManager.backgroundActor.opacity = Util.lerp(255, 0, Math.min(stateValue, 1)); - } else { - bgManager.backgroundActor.opacity = 255; - let VIGNETTE, BRIGHTNESS, bgValue; - if (opt.OVERVIEW_MODE2 && stateValue <= 1 && !opt.WORKSPACE_MODE) { - VIGNETTE = 0; - BRIGHTNESS = 1; - bgValue = stateValue; - } else { - VIGNETTE = 0.2; - BRIGHTNESS = opt.OVERVIEW_BG_BRIGHTNESS; - if (opt.OVERVIEW_MODE2 && stateValue > 1 && !opt.WORKSPACE_MODE) - bgValue = stateValue - 1; - else - bgValue = stateValue; - } - - let blurEffect = bgManager.backgroundActor.get_effect('blur'); - if (!blurEffect) { - blurEffect = new Shell.BlurEffect({ - brightness: 1, - mode: Shell.BlurMode.ACTOR, - }); - bgManager.backgroundActor.add_effect_with_name('blur', blurEffect); - } - - // In GNOME 46 the "sigma" property has been renamed to "radius" - const radius = blurEffect.sigma !== undefined ? 'sigma' : 'radius'; - - const searchActive = this._searchController.searchActive; - if (searchActive) - BRIGHTNESS = opt.SEARCH_BG_BRIGHTNESS; - - bgManager.backgroundActor.content.vignette_sharpness = VIGNETTE; - bgManager.backgroundActor.content.brightness = BRIGHTNESS; - - let vignetteInit, brightnessInit;// , sigmaInit; - if (opt.SHOW_BG_IN_OVERVIEW && opt.SHOW_WS_PREVIEW_BG) { - vignetteInit = VIGNETTE; - brightnessInit = BRIGHTNESS; - // sigmaInit = opt.OVERVIEW_BG_BLUR_SIGMA; - } else { - vignetteInit = 0; - brightnessInit = 1; - // sigmaInit = 0; - } - - if (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE) { - bgManager.backgroundActor.content.vignette_sharpness = Util.lerp(vignetteInit, VIGNETTE, bgValue); - bgManager.backgroundActor.content.brightness = Util.lerp(brightnessInit, BRIGHTNESS, bgValue); - } else { - bgManager.backgroundActor.content.vignette_sharpness = Util.lerp(vignetteInit, VIGNETTE, Math.min(stateValue, 1)); - bgManager.backgroundActor.content.brightness = Util.lerp(brightnessInit, BRIGHTNESS, Math.min(stateValue, 1)); - } - - if (opt.OVERVIEW_BG_BLUR_SIGMA || opt.APP_GRID_BG_BLUR_SIGMA) { - // reduce number of steps of blur transition to improve performance - const step = opt.SMOOTH_BLUR_TRANSITIONS ? 0.05 : 0.2; - const progress = stateValue - (stateValue % step); - if (opt.SHOW_WS_PREVIEW_BG && stateValue < 1 && !searchActive) { // no need to animate transition, unless appGrid state is involved, static bg is covered by the ws preview bg - if (blurEffect[radius] !== opt.OVERVIEW_BG_BLUR_SIGMA) - blurEffect[radius] = opt.OVERVIEW_BG_BLUR_SIGMA; - } else if (stateValue < 1 && !searchActive && !(opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE)) { - const sigma = Math.round(Util.lerp(0, opt.OVERVIEW_BG_BLUR_SIGMA, progress)); - if (sigma !== blurEffect[radius]) - blurEffect[radius] = sigma; - } else if (stateValue < 1 && !searchActive && (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE && blurEffect[radius])) { - const sigma = Math.round(Util.lerp(0, opt.OVERVIEW_BG_BLUR_SIGMA, progress)); - if (sigma !== blurEffect[radius]) - blurEffect[radius] = sigma; - } else if (stateValue > 1 && !searchActive && (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE && finalState === 1)) { - const sigma = Math.round(Util.lerp(0, opt.OVERVIEW_BG_BLUR_SIGMA, progress % 1)); - if (sigma !== blurEffect[radius]) - blurEffect[radius] = sigma; - } else if ((stateValue > 1 && bgManager._primary) || searchActive) { - const sigma = Math.round(Util.lerp(opt.OVERVIEW_BG_BLUR_SIGMA, opt.APP_GRID_BG_BLUR_SIGMA, progress % 1)); - if (sigma !== blurEffect[radius]) - blurEffect[radius] = sigma; - } else if (stateValue === 1 && !(opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE)) { - blurEffect[radius] = opt.OVERVIEW_BG_BLUR_SIGMA; - } else if (stateValue === 0 || (stateValue === 1 && (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE))) { - blurEffect[radius] = 0; - } - } - } - }, -}; - -const ControlsManagerLayoutCommon = { - after__updateWorkAreaBox() { - const workArea = this._workAreaBox.copy(); - - // opt.PANEL_OVERVIEW_ONLY removes affectsStruts panel property - if (opt.get('panelModule') && opt.PANEL_OVERVIEW_ONLY) { - let offsetY = 0; - let reduction = 0; - reduction = Main.panel.height; - offsetY = opt.PANEL_POSITION_TOP ? reduction : 0; - - const startX = workArea.x1; - const startY = workArea.y1 + offsetY; - const width = workArea.get_width(); - const height = workArea.get_height() - reduction; - - workArea.set_origin(startX, startY); - workArea.set_size(width, height); - } - - this._workAreaBoxForVShellConfig = workArea; - }, - - _updatePositionFromDashToDock() { - // update variables that cannot be processed within settings - const dash = Main.overview.dash; - opt.DASH_POSITION = dash._position; - opt.DASH_TOP = opt.DASH_POSITION === 0; - opt.DASH_RIGHT = opt.DASH_POSITION === 1; - opt.DASH_BOTTOM = opt.DASH_POSITION === 2; - opt.DASH_LEFT = opt.DASH_POSITION === 3; - opt.DASH_VERTICAL = opt.DASH_LEFT || opt.DASH_RIGHT; - }, - - _dashToDockAffectsWorkArea() { - const dash = Main.overview.dash; - const dtd = dash.get_parent()?.get_parent()?.get_parent(); - const layoutManager = Main.layoutManager; - const index = layoutManager._findActor(dtd); - const data = index > -1 ? layoutManager._trackedActors[index] : null; - const affectsStruts = data?.affectsStruts; - return !!affectsStruts; - }, -}; - -const ControlsManagerLayoutVertical = { - _computeWorkspacesBoxForState(state, box, wsTmbWidth, wsTmbHeight, leftBoxOffset, rightBoxOffset, topBoxOffset, bottomBoxOffset, centeredBoxOffset) { - const workspaceBox = box.copy(); - let [width, height] = this._workAreaBoxForVShellConfig.get_size(); - const startX = this._workAreaBoxForVShellConfig.x1; - const startY = this._workAreaBoxForVShellConfig.y1; - - let wsBoxWidth, wsBoxHeight, wsBoxY, wsBoxX; - - switch (state) { - case ControlsState.HIDDEN: - workspaceBox.set_origin(...this._workAreaBox.get_origin()); - workspaceBox.set_size(...this._workAreaBox.get_size()); - break; - case ControlsState.WINDOW_PICKER: - case ControlsState.APP_GRID: - if (opt.WS_ANIMATION && opt.SHOW_WS_TMB && state === ControlsState.APP_GRID) { - workspaceBox.set_origin(...this._workspacesThumbnails.get_position()); - workspaceBox.set_size(wsTmbWidth, wsTmbHeight); - } else if (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE) { - workspaceBox.set_origin(...this._workAreaBox.get_origin()); - workspaceBox.set_size(...this._workAreaBox.get_size()); - } else { - wsBoxWidth = width - leftBoxOffset - rightBoxOffset; - wsBoxHeight = height - topBoxOffset - bottomBoxOffset; - - const ratio = width / height; - let wRatio = wsBoxWidth / wsBoxHeight; - let scale = ratio / wRatio; - - if (scale > 1) { - wsBoxHeight /= scale; - wsBoxWidth = wsBoxHeight * ratio; - } else { - wsBoxWidth *= scale; - wsBoxHeight = wsBoxWidth / ratio; - } - - // height decides the actual size, ratio is given by the workArea - wsBoxHeight = Math.round(wsBoxHeight * opt.WS_PREVIEW_SCALE); - wsBoxWidth = Math.round(wsBoxWidth * opt.WS_PREVIEW_SCALE); - - let xOffset = 0; - let yOffset = 0; - - const yAvailableSpace = Math.round((height - topBoxOffset - wsBoxHeight - bottomBoxOffset) / 2); - yOffset = topBoxOffset + yAvailableSpace; - - const centeredBoxX = Math.round((width - wsBoxWidth) / 2); - - this._xAlignCenter = false; - if (centeredBoxX < centeredBoxOffset) { - xOffset = Math.round(leftBoxOffset + (width - leftBoxOffset - wsBoxWidth - rightBoxOffset) / 2); - } else { - xOffset = centeredBoxX; - this._xAlignCenter = true; - } - - wsBoxX = startX + xOffset; - wsBoxY = startY + yOffset; - workspaceBox.set_origin(wsBoxX, wsBoxY); - workspaceBox.set_size(wsBoxWidth, wsBoxHeight); - } - } - - return workspaceBox; - }, - - _getAppDisplayBoxForState(state, box, leftBoxOffset, rightBoxOffset, topBoxOffset, bottomBoxOffset) { - const appDisplayBox = new Clutter.ActorBox(); - const startX = this._workAreaBoxForVShellConfig.x1; - const startY = this._workAreaBoxForVShellConfig.y1; - let [width, height] = this._workAreaBoxForVShellConfig.get_size(); - const centeredBoxOffset = Math.max(leftBoxOffset, rightBoxOffset); - - const adWidth = opt.CENTER_APP_GRID - ? width - 2 * centeredBoxOffset - : width - leftBoxOffset - rightBoxOffset; - const adHeight = height - topBoxOffset - bottomBoxOffset; - - const appDisplayX = startX + - (opt.CENTER_APP_GRID - ? Math.round((width - adWidth) / 2) - : leftBoxOffset - ); - const appDisplayY = startY + topBoxOffset; - - switch (state) { - case ControlsState.HIDDEN: - case ControlsState.WINDOW_PICKER: - // 1 - left, 2 - right, 3 - bottom, 5 - top - switch (opt.APP_GRID_ANIMATION) { - case 0: - appDisplayBox.set_origin(appDisplayX, appDisplayY); - break; - case 1: - appDisplayBox.set_origin(startX + width, appDisplayY); - break; - case 2: - appDisplayBox.set_origin(box.x1 - adWidth, appDisplayY); - break; - case 3: - appDisplayBox.set_origin(appDisplayX, box.y2); - break; - case 5: - appDisplayBox.set_origin(appDisplayX, box.y1 - adHeight); - break; - } - break; - case ControlsState.APP_GRID: - appDisplayBox.set_origin(appDisplayX, appDisplayY); - break; - } - - appDisplayBox.set_size(adWidth, adHeight); - return appDisplayBox; - }, - - vfunc_allocate(container, box) { - const childBox = new Clutter.ActorBox(); - const startX = this._workAreaBoxForVShellConfig.x1; - const startY = this._workAreaBoxForVShellConfig.y1; - let [width, height] = this._workAreaBoxForVShellConfig.get_size(); - - const transitionParams = this._stateAdjustment.getStateTransitionParams(); - const spacing = opt.SPACING; - - // Dash - const maxDashHeight = Math.round(box.get_height() * DASH_MAX_SIZE_RATIO); - const maxDashWidth = Math.round(maxDashHeight * 0.8); - let dashHeight = 0; - let dashWidth = 0; - - // dash cloud be overridden by the Dash to Dock clone - if (Me.Util.dashIsDashToDock()) { - this._updatePositionFromDashToDock(); - // If DtD affects workArea, dash size needs to be 0 + spacing - const dash = Main.overview.dash; - if (this._dashToDockAffectsWorkArea()) { - if (opt.DASH_VERTICAL) - dashWidth = spacing; - else - dashHeight = spacing; - } else { - dashHeight = dash.height; - dashWidth = dash.width; - if (opt.DASH_VERTICAL) - dashWidth += spacing; - else - dashHeight += spacing; - } - } else if (this._dash.visible) { - // default dock - if (opt.DASH_VERTICAL) { - this._dash.setMaxSize(maxDashWidth, height); - [, dashWidth] = this._dash.get_preferred_width(height); - [, dashHeight] = this._dash.get_preferred_height(dashWidth); - dashWidth = Math.min(dashWidth, maxDashWidth); - dashHeight = Math.min(dashHeight, height); - } else if (!opt.WS_TMB_FULL) { - this._dash.setMaxSize(width, maxDashHeight); - [, dashHeight] = this._dash.get_preferred_height(width); - [, dashWidth] = this._dash.get_preferred_width(dashHeight); - dashHeight = Math.min(dashHeight, maxDashHeight); - dashWidth = Math.min(dashWidth, width); - } - } - - // Workspace Thumbnails - let wsTmbWidth = 0; - let wsTmbHeight = 0; - - if (opt.SHOW_WS_TMB) { - const searchActive = this._searchController.searchActive; - let maxWsTmbScale = this._dash.showAppsButton.checked && !(searchActive && !opt.SEARCH_APP_GRID_MODE) - ? opt.MAX_THUMBNAIL_SCALE_APPGRID - : opt.MAX_THUMBNAIL_SCALE; - if (transitionParams.currentState % 1 && !opt.MAX_THUMBNAIL_SCALE_STABLE && !searchActive && !opt.LEAVING_SEARCH) { - const initState = transitionParams.initialState === ControlsState.APP_GRID ? opt.MAX_THUMBNAIL_SCALE_APPGRID : opt.MAX_THUMBNAIL_SCALE; - const finalState = transitionParams.finalState === ControlsState.APP_GRID ? opt.MAX_THUMBNAIL_SCALE_APPGRID : opt.MAX_THUMBNAIL_SCALE; - maxWsTmbScale = Util.lerp(initState, finalState, transitionParams.progress); - } - wsTmbWidth = Math.round(width * maxWsTmbScale); - - let totalTmbSpacing; - [totalTmbSpacing, wsTmbHeight] = this._workspacesThumbnails.get_preferred_height(wsTmbWidth); - wsTmbHeight += totalTmbSpacing; - - const wstTopOffset = !opt.WS_TMB_FULL && opt.DASH_TOP ? dashHeight : spacing; - const wstBottomOffset = !opt.WS_TMB_FULL && opt.DASH_BOTTOM ? dashHeight : spacing; - const wstLeftOffset = opt.DASH_LEFT ? dashWidth : spacing; - const wstRightOffset = opt.DASH_RIGHT ? dashWidth : spacing; - - const wsTmbHeightMax = height - wstTopOffset - wstBottomOffset; - - // Reduce size to fit wsTmb to the screen - if (wsTmbHeight > wsTmbHeightMax) { - wsTmbHeight = wsTmbHeightMax; - wsTmbWidth = this._workspacesThumbnails.get_preferred_width(wsTmbHeight)[1]; - } - - let wsTmbX = opt.WS_TMB_LEFT - ? startX + wstLeftOffset - : startX + width - wstRightOffset - wsTmbWidth; - - let offset = (height - wstTopOffset - wsTmbHeight - wstBottomOffset) / 2; - offset = Math.round(offset - (opt.WS_TMB_POSITION_ADJUSTMENT * offset)); - const wsTmbY = startY + wstTopOffset + offset; - - childBox.set_origin(wsTmbX, wsTmbY); - childBox.set_size(Math.max(wsTmbWidth, 1), Math.max(wsTmbHeight, 1)); - - this._workspacesThumbnails.allocate(childBox); - } - - if (this._dash.visible) { - const wMaxWidth = width - spacing - wsTmbWidth - 2 * spacing - (opt.DASH_VERTICAL ? dashWidth + spacing : 0); - if (opt.WS_TMB_FULL && !opt.DASH_VERTICAL) { - this._dash.setMaxSize(wMaxWidth, maxDashHeight); - [, dashHeight] = this._dash.get_preferred_height(wMaxWidth); - [, dashWidth] = this._dash.get_preferred_width(dashHeight); - dashHeight = Math.min(dashHeight, maxDashHeight); - dashWidth = Math.min(dashWidth, wMaxWidth); - } - - let dashX = opt.DASH_RIGHT ? width - dashWidth : 0; - let dashY = opt.DASH_TOP ? startY : startY + height - dashHeight; - - if (!opt.DASH_VERTICAL) { - const dashLeftOffset = (opt.WS_TMB_FULL || opt.CENTER_DASH_WS) && opt.WS_TMB_LEFT ? wsTmbWidth + spacing : 0; - const dashRightOffset = (opt.WS_TMB_FULL || opt.CENTER_DASH_WS) && opt.WS_TMB_RIGHT ? wsTmbWidth + spacing : 0; - let offset = (width - dashWidth - (opt.CENTER_DASH_WS && !this._xAlignCenter ? dashLeftOffset + dashRightOffset : 0)) / 2; - offset -= opt.DASH_POSITION_ADJUSTMENT * (offset - spacing); - dashX = startX + (opt.CENTER_DASH_WS ? dashLeftOffset : 0) + offset; - if (opt.WS_TMB_FULL) // Limit the adjustment while keeping the center of adjustment on the screen center - dashX = Math.clamp(startX + dashLeftOffset + spacing, dashX, startX + width - dashRightOffset - spacing - dashWidth); - } else { - const offset = (height - dashHeight) / 2; - dashY = startY + (offset - opt.DASH_POSITION_ADJUSTMENT * (offset - spacing)); - } - dashY = Math.round(dashY); - - childBox.set_origin(startX + dashX, dashY); - childBox.set_size(dashWidth, dashHeight); - this._dash.allocate(childBox); - } - - // View box offsets - const leftBoxOffset = (opt.DASH_LEFT ? dashWidth : spacing) + (opt.WS_TMB_LEFT ? wsTmbWidth + spacing : 0); - const rightBoxOffset = (opt.DASH_RIGHT ? dashWidth : spacing) + (opt.WS_TMB_RIGHT ? wsTmbWidth + spacing : 0); - let topBoxOffset = (opt.DASH_TOP ? dashHeight : spacing) + (opt.WS_TMB_TOP ? wsTmbHeight + spacing : 0); - const bottomBoxOffset = (opt.DASH_BOTTOM ? dashHeight : spacing) + (opt.WS_TMB_BOTTOM ? wsTmbHeight + spacing : 0); - const centeredBoxOffset = Math.max(leftBoxOffset, rightBoxOffset); - - // App grid needs to be calculated for the max wsTmbWidth in app grid, independently on the current wsTmb scale - const wsTmbWidthAppGrid = Math.round(width * opt.MAX_THUMBNAIL_SCALE_APPGRID); - const leftBoxOffsetAppGrid = (opt.DASH_LEFT ? dashWidth : spacing) + (opt.WS_TMB_LEFT ? wsTmbWidthAppGrid + spacing : 0); - const rightBoxOffsetAppGrid = (opt.DASH_RIGHT ? dashWidth : spacing) + (opt.WS_TMB_RIGHT ? wsTmbWidthAppGrid + spacing : 0); - - // searchEntry - const [searchEntryHeight] = this._searchEntry.get_preferred_height(width - wsTmbWidth); - const searchEntryY = startY + topBoxOffset; - - const searchX = startX + - (opt.CENTER_SEARCH_VIEW || this._xAlignCenter - ? centeredBoxOffset - : leftBoxOffset); // xAlignCenter is set by wsBox - - const searchWidth = - width - (opt.CENTER_SEARCH_VIEW || this._xAlignCenter - ? 2 * centeredBoxOffset - : leftBoxOffset + rightBoxOffset); - - childBox.set_origin(searchX, searchEntryY); - childBox.set_size(searchWidth, searchEntryHeight); - - this._searchEntry.allocate(childBox); - - // searchResults - const searchY = startY + topBoxOffset + searchEntryHeight + spacing; - const searchHeight = height - topBoxOffset - bottomBoxOffset - searchEntryHeight - 2 * spacing; - - childBox.set_origin(searchX, searchY); - childBox.set_size(searchWidth, searchHeight); - this._searchController.allocate(childBox); - - // Add searchEntry height if needed - topBoxOffset += opt.SHOW_SEARCH_ENTRY ? searchEntryHeight + spacing : 0; - - // workspace - let params = [box, wsTmbWidth, wsTmbHeight, leftBoxOffset, rightBoxOffset, topBoxOffset, bottomBoxOffset, centeredBoxOffset]; - - // Update cached boxes - for (const state of Object.values(ControlsState)) { - this._cachedWorkspaceBoxes.set( - state, this._computeWorkspacesBoxForState(state, ...params)); - } - - let workspacesBox; - if (!transitionParams.transitioning) - workspacesBox = this._cachedWorkspaceBoxes.get(transitionParams.currentState); - - if (!workspacesBox) { - const initialBox = this._cachedWorkspaceBoxes.get(transitionParams.initialState); - const finalBox = this._cachedWorkspaceBoxes.get(transitionParams.finalState); - workspacesBox = initialBox.interpolate(finalBox, transitionParams.progress); - } - - this._workspacesDisplay.allocate(workspacesBox); - - // appDisplay - params = [ - box, - leftBoxOffsetAppGrid, - rightBoxOffsetAppGrid, - topBoxOffset, - bottomBoxOffset, - ]; - let appDisplayBox; - if (!transitionParams.transitioning) { - appDisplayBox = - this._getAppDisplayBoxForState(transitionParams.currentState, ...params); - } else { - const initialBox = - this._getAppDisplayBoxForState(transitionParams.initialState, ...params); - const finalBox = - this._getAppDisplayBoxForState(transitionParams.finalState, ...params); - - appDisplayBox = initialBox.interpolate(finalBox, transitionParams.progress); - } - this._appDisplay.allocate(appDisplayBox); - - this._runPostAllocation(); - }, -}; - -const ControlsManagerLayoutHorizontal = { - _computeWorkspacesBoxForState: ControlsManagerLayoutVertical._computeWorkspacesBoxForState, - - _getAppDisplayBoxForState: ControlsManagerLayoutVertical._getAppDisplayBoxForState, - - vfunc_allocate(container, box) { - const childBox = new Clutter.ActorBox(); - const startX = this._workAreaBoxForVShellConfig.x1; - const startY = this._workAreaBoxForVShellConfig.y1; - let [width, height] = this._workAreaBoxForVShellConfig.get_size(); - - const transitionParams = this._stateAdjustment.getStateTransitionParams(); - const spacing = opt.SPACING; - - // Dash - const maxDashHeight = Math.round(box.get_height() * DASH_MAX_SIZE_RATIO); - const maxDashWidth = Math.round(maxDashHeight * 0.8); - let dashHeight = 0; - let dashWidth = 0; - - // dash cloud be overridden by the Dash to Dock clone - if (Me.Util.dashIsDashToDock()) { - this._updatePositionFromDashToDock(); - // If DtD affects workArea, dash size needs to be 0 - const dash = Main.overview.dash; - if (this._dashToDockAffectsWorkArea()) { - if (opt.DASH_VERTICAL) - dashWidth = spacing; - else - dashHeight = spacing; - } else { - dashHeight = dash.height; - dashWidth = dash.width; - if (opt.DASH_VERTICAL) - dashWidth += spacing; - else - dashHeight += spacing; - } - } else if (this._dash.visible) { - // default dock - if (!opt.DASH_VERTICAL) { - this._dash.setMaxSize(width, maxDashHeight); - [, dashHeight] = this._dash.get_preferred_height(width); - [, dashWidth] = this._dash.get_preferred_width(dashHeight); - dashHeight = Math.min(dashHeight, maxDashHeight); - dashWidth = Math.min(dashWidth, width - spacing); - } else if (!opt.WS_TMB_FULL) { - this._dash.setMaxSize(maxDashWidth, height); - [, dashWidth] = this._dash.get_preferred_width(height); - [, dashHeight] = this._dash.get_preferred_height(dashWidth); - dashHeight = Math.min(dashHeight, height - spacing); - dashWidth = Math.min(dashWidth, width); - } - } - - const [searchEntryHeight] = this._searchEntry.get_preferred_height(width); - - // Workspace Thumbnails - let wsTmbWidth = 0; - let wsTmbHeight = 0; - - if (opt.SHOW_WS_TMB) { - const searchActive = this._searchController.searchActive; - let maxWsTmbScale = this._dash.showAppsButton.checked && !(searchActive && !opt.SEARCH_APP_GRID_MODE) - ? opt.MAX_THUMBNAIL_SCALE_APPGRID - : opt.MAX_THUMBNAIL_SCALE; - if (transitionParams.currentState % 1 && !opt.MAX_THUMBNAIL_SCALE_STABLE && !searchActive && !opt.LEAVING_SEARCH) { - const initState = transitionParams.initialState === ControlsState.APP_GRID ? opt.MAX_THUMBNAIL_SCALE_APPGRID : opt.MAX_THUMBNAIL_SCALE; - const finalState = transitionParams.finalState === ControlsState.APP_GRID ? opt.MAX_THUMBNAIL_SCALE_APPGRID : opt.MAX_THUMBNAIL_SCALE; - maxWsTmbScale = Util.lerp(initState, finalState, transitionParams.progress); - } - - wsTmbHeight = Math.round(height * maxWsTmbScale); - - let totalTmbSpacing; - [totalTmbSpacing, wsTmbWidth] = this._workspacesThumbnails.get_preferred_width(wsTmbHeight); - wsTmbWidth += totalTmbSpacing; - - const wstLeftOffset = !opt.WS_TMB_FULL && opt.DASH_LEFT ? dashWidth : spacing; - const wstRightOffset = !opt.WS_TMB_FULL && opt.DASH_RIGHT ? dashWidth : spacing; - const wstTopOffset = opt.DASH_TOP ? dashHeight : spacing; - const wstBottomOffset = opt.DASH_BOTTOM ? dashHeight : spacing; - - const wsTmbWidthMax = width - wstLeftOffset - wstRightOffset; - // Reduce size to fit wsTmb to the screen - if (wsTmbWidth > wsTmbWidthMax) { - wsTmbWidth = wsTmbWidthMax; - wsTmbHeight = this._workspacesThumbnails.get_preferred_height(wsTmbWidth)[1]; - } - - let wsTmbY = opt.WS_TMB_TOP - ? startY + wstTopOffset - : startY + height - wstBottomOffset - wsTmbHeight; - - let offset = (width - wstLeftOffset - wsTmbWidth - wstRightOffset) / 2; - offset = Math.round(offset - (opt.WS_TMB_POSITION_ADJUSTMENT * offset)); - const wsTmbX = startX + wstLeftOffset + offset; - - childBox.set_origin(wsTmbX, wsTmbY); - childBox.set_size(Math.max(wsTmbWidth, 1), Math.max(wsTmbHeight, 1)); - this._workspacesThumbnails.allocate(childBox); - } - - if (this._dash.visible) { - if (opt.WS_TMB_FULL && opt.DASH_VERTICAL) { - const wMaxHeight = height - spacing - wsTmbHeight; - this._dash.setMaxSize(maxDashWidth, wMaxHeight); - [, dashWidth] = this._dash.get_preferred_width(wMaxHeight); - [, dashHeight] = this._dash.get_preferred_height(dashWidth); - dashWidth = Math.min(dashWidth, maxDashWidth); - dashHeight = Math.min(dashHeight, wMaxHeight); - } - - let dashX = opt.DASH_RIGHT ? width - dashWidth : 0; - let dashY = opt.DASH_TOP ? startY : startY + height - dashHeight; - - if (opt.DASH_VERTICAL) { - const dashTopOffset = (opt.WS_TMB_FULL || opt.CENTER_DASH_WS) && opt.WS_TMB_TOP ? wsTmbHeight + spacing : 0; - const dashBottomOffset = (opt.WS_TMB_FULL || opt.CENTER_DASH_WS) && opt.WS_TMB_BOTTOM ? wsTmbHeight + spacing : 0; - let offset = (height - dashHeight - (opt.CENTER_DASH_WS ? dashTopOffset + dashBottomOffset : 0)) / 2; - offset -= opt.DASH_POSITION_ADJUSTMENT * (offset - spacing); - dashY = startY + (opt.CENTER_DASH_WS ? dashTopOffset : 0) + offset; - if (opt.WS_TMB_FULL) // Limit the adjustment while keeping the center of adjustment on the screen center - dashY = Math.clamp(startY + dashTopOffset + spacing, dashY, startY + height - dashBottomOffset - spacing - dashHeight); - } else { - const offset = (width - dashWidth) / 2; - dashX = startX + (offset - opt.DASH_POSITION_ADJUSTMENT * (offset - spacing)); - } - dashX = Math.round(dashX); - - childBox.set_origin(startX + dashX, dashY); - childBox.set_size(dashWidth, dashHeight); - this._dash.allocate(childBox); - } - - // Main view offsets - const leftBoxOffset = opt.DASH_LEFT ? dashWidth : spacing; - const rightBoxOffset = opt.DASH_RIGHT ? dashWidth : spacing; - let topBoxOffset = (opt.DASH_TOP ? dashHeight : spacing) + (opt.WS_TMB_TOP ? wsTmbHeight + spacing : 0); - const bottomBoxOffset = (opt.DASH_BOTTOM ? dashHeight : spacing) + (opt.WS_TMB_BOTTOM ? wsTmbHeight + spacing : 0); - const centeredBoxOffset = Math.max(leftBoxOffset, rightBoxOffset); - - // App grid needs to be calculated for the max wsTmbWidth in app grid, independently on the current wsTmb scale - const wsTmbHeightAppGrid = Math.round(height * opt.MAX_THUMBNAIL_SCALE_APPGRID); - const topBoxOffsetAppGrid = (opt.DASH_TOP ? dashHeight : spacing) + (opt.WS_TMB_TOP ? wsTmbHeightAppGrid + spacing : 0) + (opt.SHOW_SEARCH_ENTRY ? searchEntryHeight + spacing : 0); - const bottomBoxOffsetAppGrid = (opt.DASH_BOTTOM ? dashHeight : spacing) + (opt.WS_TMB_BOTTOM ? wsTmbHeightAppGrid + spacing : 0); - - // searchEntry - const searchEntryY = startY + topBoxOffset; - - const searchX = startX + - (opt.CENTER_SEARCH_VIEW || this._xAlignCenter - ? centeredBoxOffset - : leftBoxOffset); // xAlignCenter is set by wsBox - - const searchWidth = - width - (opt.CENTER_SEARCH_VIEW || this._xAlignCenter - ? 2 * centeredBoxOffset - : leftBoxOffset + rightBoxOffset); - - childBox.set_origin(searchX, searchEntryY); - childBox.set_size(searchWidth, searchEntryHeight); - - this._searchEntry.allocate(childBox); - - // searchResults - const searchY = startY + topBoxOffset + searchEntryHeight + spacing; - const searchHeight = height - topBoxOffset - bottomBoxOffset - searchEntryHeight - 2 * spacing; - - childBox.set_origin(searchX, searchY); - childBox.set_size(searchWidth, searchHeight); - this._searchController.allocate(childBox); - - // Add searchEntry height if needed - topBoxOffset += opt.SHOW_SEARCH_ENTRY ? searchEntryHeight + spacing : 0; - - // Workspace - let params = [ - box, - wsTmbWidth, - wsTmbHeight, - leftBoxOffset, - rightBoxOffset, - topBoxOffset, - bottomBoxOffset, - centeredBoxOffset, - ]; - - // Update cached boxes - for (const state of Object.values(ControlsState)) { - this._cachedWorkspaceBoxes.set( - state, this._computeWorkspacesBoxForState(state, ...params)); - } - - let workspacesBox; - if (!transitionParams.transitioning) - workspacesBox = this._cachedWorkspaceBoxes.get(transitionParams.currentState); - - if (!workspacesBox) { - const initialBox = this._cachedWorkspaceBoxes.get(transitionParams.initialState); - const finalBox = this._cachedWorkspaceBoxes.get(transitionParams.finalState); - workspacesBox = initialBox.interpolate(finalBox, transitionParams.progress); - } - - this._workspacesDisplay.allocate(workspacesBox); - - // appDisplay - params = [ - box, - leftBoxOffset === spacing ? 0 : leftBoxOffset, - rightBoxOffset === spacing ? 0 : rightBoxOffset, - topBoxOffsetAppGrid, - bottomBoxOffsetAppGrid, - ]; - let appDisplayBox; - if (!transitionParams.transitioning) { - appDisplayBox = - this._getAppDisplayBoxForState(transitionParams.currentState, ...params); - } else { - const initialBox = - this._getAppDisplayBoxForState(transitionParams.initialState, ...params); - const finalBox = - this._getAppDisplayBoxForState(transitionParams.finalState, ...params); - - appDisplayBox = initialBox.interpolate(finalBox, transitionParams.progress); - } - this._appDisplay.allocate(appDisplayBox); - - this._runPostAllocation(); - }, -}; - -// same copy of this function should be available in OverviewControls and WorkspacesView -function _getFitModeForState(state) { - switch (state) { - case ControlsState.HIDDEN: - case ControlsState.WINDOW_PICKER: - return FitMode.SINGLE; - case ControlsState.APP_GRID: - if (opt.WS_ANIMATION && opt.SHOW_WS_TMB) - return FitMode.ALL; - else - return FitMode.SINGLE; - default: - return FitMode.SINGLE; - } -} diff --git a/extensions/46/vertical-workspaces/lib/panel.js b/extensions/46/vertical-workspaces/lib/panel.js deleted file mode 100644 index ba6d01a..0000000 --- a/extensions/46/vertical-workspaces/lib/panel.js +++ /dev/null @@ -1,253 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * panel.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Overview from 'resource:///org/gnome/shell/ui/overview.js'; - -let Me; -let opt; - -const ANIMATION_TIME = Overview.ANIMATION_TIME; - -export const PanelModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - - this._showingOverviewConId = 0; - this._hidingOverviewConId = 0; - this._styleChangedConId = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('panelModule'); - const conflict = Me.Util.getEnabledExtensions('dash-to-panel').length || - Me.Util.getEnabledExtensions('hidetopbar').length; - - if (conflict && !reset) - console.warn(`[${Me.metadata.name}] Warning: "Panel" module disabled due to potential conflict with another extension`); - - reset = reset || !this.moduleEnabled || conflict; - - this.moduleEnabled = !reset; - - // don't touch original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' PanelModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - const panelBox = Main.layoutManager.panelBox; - - this._setPanelPosition(); - this._updateStyleChangedConnection(); - - if (!opt.PANEL_MODE) { - this._updateOverviewConnection(true); - this._reparentPanel(false); - panelBox.translation_y = 0; - Main.panel.opacity = 255; - this._setPanelStructs(true); - } else if (opt.PANEL_OVERVIEW_ONLY) { - if (opt.SHOW_WS_PREVIEW_BG) { - this._reparentPanel(true); - if (opt.OVERVIEW_MODE2) { - // in OM2 if the panel has been moved to the overviewGroup move panel above all - Main.layoutManager.overviewGroup.set_child_above_sibling(panelBox, null); - this._updateOverviewConnection(); - } else { - this._updateOverviewConnection(true); - } - this._showPanel(true); - } else { - // if ws preview bg is disabled, panel can stay in uiGroup - this._reparentPanel(false); - this._showPanel(false); - this._updateOverviewConnection(); - } - // _connectPanel(); - } else if (opt.PANEL_DISABLED) { - this._updateOverviewConnection(true); - this._reparentPanel(false); - this._showPanel(false); - // _connectPanel(); - } - this._setPanelStructs(!opt.PANEL_MODE); - Main.layoutManager._updateHotCorners(); - - this._overrides.addOverride('ActivitiesButton', Main.panel.statusArea.activities, ActivitiesButton); - - console.debug(' PanelModule - Activated'); - } - - _disableModule() { - const reset = true; - this._setPanelPosition(reset); - this._updateOverviewConnection(reset); - this._reparentPanel(false); - - this._updateStyleChangedConnection(reset); - - const panelBox = Main.layoutManager.panelBox; - panelBox.translation_y = 0; - Main.panel.opacity = 255; - this._setPanelStructs(true); - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - console.debug(' PanelModule - Disabled'); - } - - _setPanelPosition(reset = false) { - const geometry = global.display.get_monitor_geometry(global.display.get_primary_monitor()); - const panelBox = Main.layoutManager.panelBox; - const panelHeight = Main.panel.height; // panelBox height can be 0 after shell start - - if (opt.PANEL_POSITION_TOP || reset) - panelBox.set_position(geometry.x, geometry.y); - else - panelBox.set_position(geometry.x, geometry.y + geometry.height - panelHeight); - } - - _updateStyleChangedConnection(reset = false) { - if (reset) { - if (this._styleChangedConId) { - Main.panel.disconnect(this._styleChangedConId); - this._styleChangedConId = 0; - } - } else if (!this._styleChangedConId) { - this._styleChangedConId = Main.panel.connect('style-changed', () => { - if (opt.PANEL_OVERVIEW_ONLY && !opt.OVERVIEW_MODE2) - Main.panel.add_style_pseudo_class('overview'); - else if (opt.OVERVIEW_MODE2) - Main.panel.remove_style_pseudo_class('overview'); - }); - } - } - - _updateOverviewConnection(reset = false) { - if (reset) { - if (this._hidingOverviewConId) { - Main.overview.disconnect(this._hidingOverviewConId); - this._hidingOverviewConId = 0; - } - if (this._showingOverviewConId) { - Main.overview.disconnect(this._showingOverviewConId); - this._showingOverviewConId = 0; - } - } else { - if (!this._hidingOverviewConId) { - this._hidingOverviewConId = Main.overview.connect('hiding', () => { - if (!opt.SHOW_WS_PREVIEW_BG || opt.OVERVIEW_MODE2) - this._showPanel(false); - }); - } - if (!this._showingOverviewConId) { - this._showingOverviewConId = Main.overview.connect('showing', () => { - if (Main.layoutManager._startingUp) - return; - if (!opt.SHOW_WS_PREVIEW_BG || opt.OVERVIEW_MODE2 || Main.layoutManager.panelBox.translation_y) - this._showPanel(true); - }); - } - } - } - - _reparentPanel(reparent = false) { - const panel = Main.layoutManager.panelBox; - if (reparent && panel.get_parent() === Main.layoutManager.uiGroup && !Main.sessionMode.isLocked) { - Main.layoutManager.uiGroup.remove_child(panel); - Main.layoutManager.overviewGroup.add_child(panel); - } else if ((!reparent || Main.sessionMode.isLocked) && panel.get_parent() === Main.layoutManager.overviewGroup) { - Main.layoutManager.overviewGroup.remove_child(panel); - // return the panel at default position, panel shouldn't cover objects that should be above - Main.layoutManager.uiGroup.insert_child_at_index(panel, 4); - } - } - - _setPanelStructs(state) { - Main.layoutManager._trackedActors.forEach(a => { - if (a.actor === Main.layoutManager.panelBox) - a.affectsStruts = state; - }); - - // workaround to force maximized windows to resize after removing affectsStruts - // simulation of minimal swipe gesture to the opposite direction - // todo - needs better solution!!!!!!!!!!! - // const direction = _getAppGridAnimationDirection() === 2 ? 1 : -1; - // Main.overview._swipeTracker._beginTouchSwipe(null, global.get_current_time(), 1, 1); - // Main.overview._swipeTracker._updateGesture(null, global.get_current_time(), direction, 1); - // GLib.timeout_add(0, 50, () => Main.overview._swipeTracker._endGesture(global.get_current_time(), 1, true));*/ - } - - _showPanel(show = true) { - if (show) { - Main.panel.opacity = 255; - Main.layoutManager.panelBox.ease({ - duration: ANIMATION_TIME, - translation_y: 0, - onComplete: () => { - this._setPanelStructs(!opt.PANEL_MODE); - }, - }); - } else if (!Main.layoutManager._startingUp) { - const panelHeight = Main.panel.height; - Main.layoutManager.panelBox.ease({ - duration: ANIMATION_TIME, - translation_y: opt.PANEL_POSITION_TOP ? -panelHeight + 1 : panelHeight - 1, - onComplete: () => { - Main.panel.opacity = 0; - this._setPanelStructs(!opt.PANEL_MODE); - }, - }); - } - } -}; - -const ActivitiesButton = { - vfunc_event(event) { - if (event.type() === Clutter.EventType.TOUCH_END || - event.type() === Clutter.EventType.BUTTON_RELEASE) { - if (Main.overview.shouldToggleByCornerOrButton()) { - if (event.get_button() === Clutter.BUTTON_SECONDARY && !Main.overview.dash.showAppsButton.checked) { - Main.overview.show(2); - Main.overview.dash.showAppsButton.checked = true; - } else { - Main.overview.toggle(); - } - } - } - - return Main.wm.handleWorkspaceScroll(event); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/recentFilesSearchProvider.js b/extensions/46/vertical-workspaces/lib/recentFilesSearchProvider.js deleted file mode 100644 index f050cf9..0000000 --- a/extensions/46/vertical-workspaces/lib/recentFilesSearchProvider.js +++ /dev/null @@ -1,316 +0,0 @@ -/** - * Vertical Workspaces - * recentFilesSearchProvider.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - */ - -'use strict'; - -import GLib from 'gi://GLib'; -import St from 'gi://St'; -import Gio from 'gi://Gio'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; - -let Me; -let opt; -// gettext -let _; - -// prefix helps to eliminate results from other search providers -// so it needs to be something less common -// needs to be accessible from vw module -export const PREFIX = 'fq//'; -const ID = 'recent-files'; - -export const RecentFilesSearchProviderModule = class { - // export for other modules - static _PREFIX = PREFIX; - constructor(me) { - Me = me; - opt = Me.opt; - _ = Me.gettext; - - this._firstActivation = true; - this.moduleEnabled = false; - this._recentFilesSearchProvider = null; - this._enableTimeoutId = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - _ = null; - } - - update(reset) { - this.moduleEnabled = opt.get('recentFilesSearchProviderModule'); - - reset = reset || !this.moduleEnabled; - - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' RecentFilesSearchProviderModule - Keeping untouched'); - } - - _activateModule() { - // delay because Fedora had problem to register a new provider soon after Shell restarts - this._enableTimeoutId = GLib.timeout_add( - GLib.PRIORITY_DEFAULT, - 2000, - () => { - if (!this._recentFilesSearchProvider) { - this._recentFilesSearchProvider = new RecentFilesSearchProvider(); - this._registerProvider(this._recentFilesSearchProvider); - } - this._enableTimeoutId = 0; - return GLib.SOURCE_REMOVE; - } - ); - - console.debug(' RecentFilesSearchProviderModule - Activated'); - } - - _disableModule() { - if (this._recentFilesSearchProvider) { - this._unregisterProvider(this._recentFilesSearchProvider); - this._recentFilesSearchProvider = null; - } - if (this._enableTimeoutId) { - GLib.source_remove(this._enableTimeoutId); - this._enableTimeoutId = 0; - } - - console.debug(' RecentFilesSearchProviderModule - Disabled'); - } - - _registerProvider(provider) { - const searchResults = Main.overview.searchController._searchResults; - provider.searchInProgress = false; - - searchResults._providers.push(provider); - - // create results display and add it to the _content - searchResults._ensureProviderDisplay.bind(searchResults)(provider); - } - - _unregisterProvider(provider) { - const searchResults = Main.overview.searchController._searchResults; - searchResults._unregisterProvider(provider); - } -}; - -class RecentFilesSearchProvider { - constructor() { - this.id = ID; - const appId = 'org.gnome.Nautilus.desktop'; - - // A real appInfo created from a commandline has often issues with overriding get_id() method, so we use dict instead - this.appInfo = { - get_id: () => appId, - get_name: () => _('Recent Files'), - get_icon: () => Gio.icon_new_for_string('focus-windows-symbolic'), - should_show: () => true, - get_commandline: () => '/usr/bin/nautilus -w recent:///', - launch: () => {}, - }; - - this.canLaunchSearch = true; - this.isRemoteProvider = false; - - this._recentFilesManager = new RecentFilesManager(); - } - - getInitialResultSet(terms/* , cancellable*/) { - const rfm = this._recentFilesManager; - rfm.loadFromFile(); - - const uris = rfm.getUris(); - const dict = {}; - for (let uri of uris) { - dict[uri] = {}; - dict[uri]['uri'] = uri; - dict[uri]['path'] = rfm.getPath(uri); - dict[uri]['filename'] = rfm.getDisplayName(uri); - dict[uri]['dir'] = rfm.getDirPath(uri); - dict[uri]['age'] = rfm.getAge(uri); - dict[uri]['appInfo'] = rfm.getDefaultAppAppInfo(uri); - } - this.files = dict; - - return new Promise(resolve => resolve(this._getResultSet(terms))); - } - - _getResultSet(terms) { - if (!terms[0].startsWith(PREFIX)) - return []; - // do not modify original terms - let termsCopy = [...terms]; - // search for terms without prefix - termsCopy[0] = termsCopy[0].replace(PREFIX, ''); - - const candidates = Object.values(this.files); - const _terms = [].concat(termsCopy); - - const term = _terms.join(' '); - - const results = []; - let m; - for (let file of candidates) { - if (opt.SEARCH_FUZZY) - m = Me.Util.fuzzyMatch(term, file.filename); - else - m = Me.Util.strictMatch(term, file.filename); - - if (m !== -1) - results.push(file); - } - - results.sort((a, b) => a.age > b.age); - - const resultIds = results.map(item => item.uri); - return resultIds; - } - - getResultMetas(resultIds/* , callback = null*/) { - const metas = resultIds.map(id => this.getResultMeta(id)); - return new Promise(resolve => resolve(metas)); - } - - getResultMeta(resultId) { - const result = this.files[resultId]; - return { - 'id': resultId, - 'name': `${Math.floor(result.age)}: ${result.filename}`, - 'description': `${result.dir}`, - 'createIcon': size => - this._recentFilesManager.getDefaultAppIcon(resultId, size), - }; - } - - launchSearch(terms, timeStamp) { - const appInfo = Gio.AppInfo.create_from_commandline('/usr/bin/nautilus -w recent:///', 'Nautilus', null); - appInfo.launch([], global.create_app_launch_context(timeStamp, -1)); - - // unlike on 42, on 44 if a window with the same uri is already open it will not get focus/activation - // Gio.app_info_launch_default_for_uri('recent:///', global.create_app_launch_context(timeStamp, -1)); - - // following solution for some reason ignores the recent:/// uri - // this.appInfo.launch_uris(['recent:///'], global.create_app_launch_context(timeStamp, -1)); - } - - activateResult(resultId, terms, timeStamp) { - const uri = resultId; - const context = global.create_app_launch_context(timeStamp, -1); - if (Me.Util.isShiftPressed()) { - Main.overview.toggle(); - this.appInfo.launch_uris([uri], context); - } else if (Gio.app_info_launch_default_for_uri(uri, context)) { - // update recent list after successful activation - this._recentFilesManager.updateAdded(resultId); - this._recentFilesManager.saveToFile(); - } else { - this.appInfo.launch_uris([uri], context); - } - } - - filterResults(results /* , maxResults*/) { - // return results.slice(0, maxResults); - return results.slice(0, 20); - } - - getSubsearchResultSet(previousResults, terms/* , cancellable*/) { - return this.getInitialResultSet(terms); - } -} - -class RecentFilesManager { - constructor(path) { - path = path ?? GLib.build_filenamev([GLib.get_user_data_dir(), 'recently-used.xbel']); - this._recentlyUsedPath = path; - this._bookmarks = new GLib.BookmarkFile(); - } - - loadFromFile() { - try { - this._bookmarks.load_from_file(this._recentlyUsedPath); - } catch (e) { - if (!e.matches(GLib.BookmarkFileError, GLib.BookmarkFileError.FILE_NOT_FOUND)) - console.error(`Could not open recent files: ${e.message}`); - } - } - - saveToFile() { - try { - this._bookmarks.to_file(this._recentlyUsedPath); - } catch (e) { - if (!e.matches(GLib.BookmarkFileError, GLib.BookmarkFileError.FILE_NOT_FOUND)) - console.error(`Could not open recent files to save data: ${e.message}`); - } - } - - getUris() { - return this._bookmarks.get_uris(); - } - - getPath(uri) { - // GLib.filename_from_uri() removes uri schema and converts string to utf-8 - return GLib.filename_from_uri(uri)[0]; // result is array - } - - getDisplayName(uri) { - const path = this.getPath(uri); - return GLib.filename_display_basename(path); - } - - getDirPath(uri) { - const path = this.getPath(uri); - const filename = this.getDisplayName(uri); - return path.replace(`${filename}`, ''); - } - - getMimeType(uri) { - return this._bookmarks.get_mime_type(uri); - } - - getAdded(uri) { - return this._bookmarks.get_added(uri); - } - - updateAdded(uri) { - this._bookmarks.set_added_date_time(uri, GLib.DateTime.new_now_local()); - } - - // age in days (float) - getAge(uri) { - return (Date.now() / 1000 - this._bookmarks.get_added(uri)) / 60 / 60 / 24; - } - - getDefaultAppAppInfo(uri) { - const mimeType = this.getMimeType(uri); - return Gio.AppInfo.get_default_for_type(mimeType, false); - } - - getDefaultAppIcon(uri, size) { - let icon, gicon; - - const appInfo = this.getDefaultAppAppInfo(uri); - if (appInfo) - gicon = appInfo.get_icon(); - - if (gicon) - icon = new St.Icon({ gicon, icon_size: size }); - else - icon = new St.Icon({ icon_name: 'icon-missing', icon_size: size }); - - return icon; - } -} diff --git a/extensions/46/vertical-workspaces/lib/search.js b/extensions/46/vertical-workspaces/lib/search.js deleted file mode 100644 index 47198a7..0000000 --- a/extensions/46/vertical-workspaces/lib/search.js +++ /dev/null @@ -1,474 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * search.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import GLib from 'gi://GLib'; -import Clutter from 'gi://Clutter'; -import St from 'gi://St'; -import Shell from 'gi://Shell'; -import GObject from 'gi://GObject'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Search from 'resource:///org/gnome/shell/ui/search.js'; -import * as AppDisplay from 'resource:///org/gnome/shell/ui/appDisplay.js'; - -import * as SystemActions from 'resource:///org/gnome/shell/misc/systemActions.js'; -import { Highlighter } from 'resource:///org/gnome/shell/misc/util.js'; - -let Me; -// gettext -let _; -let opt; - -const SEARCH_MAX_WIDTH = 1092; - -export const SearchModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - _ = Me.gettext; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - _ = null; - } - - update(reset) { - this.moduleEnabled = opt.get('searchModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' SearchModule - Keeping untouched'); - } - - _activateModule() { - this._updateSearchViewWidth(); - - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('AppSearchProvider', AppDisplay.AppSearchProvider.prototype, AppSearchProvider); - this._overrides.addOverride('SearchResult', Search.SearchResult.prototype, SearchResult); - this._overrides.addOverride('SearchResultsView', Search.SearchResultsView.prototype, SearchResultsView); - this._overrides.addOverride('ListSearchResults', Search.ListSearchResults.prototype, ListSearchResults); - this._overrides.addOverride('ListSearchResult', Search.ListSearchResult.prototype, ListSearchResultOverride); - this._overrides.addOverride('Highlighter', Highlighter.prototype, HighlighterOverride); - - // Don't expand the search view vertically and align it to the top - // this is important in the static workspace mode when the search view bg is not transparent - // also the "Searching..." and "No Results" notifications will be closer to the search entry, with the distance given by margin-top in the stylesheet - Main.overview.searchController.y_align = Clutter.ActorAlign.START; - // Increase the maxResults for app search so that it can show more results in case the user decreases the size of the result icon - const appSearchDisplay = Main.overview.searchController._searchResults._providers.filter(p => p.id === 'applications')[0]?.display; - if (appSearchDisplay) - appSearchDisplay._maxResults = 12; - console.debug(' SearchModule - Activated'); - } - - _disableModule() { - const reset = true; - - const searchResults = Main.overview.searchController._searchResults; - if (searchResults?._searchTimeoutId) { - GLib.source_remove(searchResults._searchTimeoutId); - searchResults._searchTimeoutId = 0; - } - - this._updateSearchViewWidth(reset); - - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - Main.overview.searchController.y_align = Clutter.ActorAlign.FILL; - - console.debug(' WorkspaceSwitcherPopupModule - Disabled'); - } - - _updateSearchViewWidth(reset = false) { - const searchContent = Main.overview.searchController._searchResults._content; - - if (reset) { - searchContent.set_style(''); - } else { - let width = SEARCH_MAX_WIDTH; - if (Me.Util.monitorHasLowResolution()) - width = Math.round(width * 0.8); - width = Math.round(width * opt.SEARCH_VIEW_SCALE); - searchContent.set_style(`max-width: ${width}px;`); - } - } -}; - -const ListSearchResults = { - _getMaxDisplayedResults() { - return opt.SEARCH_MAX_ROWS; - }, -}; - -// AppDisplay.AppSearchProvider -const AppSearchProvider = { - getInitialResultSet(terms, cancellable) { - // Defer until the parental controls manager is initialized, so the - // results can be filtered correctly. - if (!this._parentalControlsManager.initialized) { - return new Promise(resolve => { - let initializedId = this._parentalControlsManager.connect('app-filter-changed', async () => { - if (this._parentalControlsManager.initialized) { - this._parentalControlsManager.disconnect(initializedId); - resolve(await this.getInitialResultSet(terms, cancellable)); - } - }); - }); - } - - const pattern = terms.join(' '); - - let appInfoList = Shell.AppSystem.get_default().get_installed(); - - let weightList = {}; - appInfoList = appInfoList.filter(appInfo => { - try { - appInfo.get_id(); // catch invalid file encodings - } catch (e) { - return false; - } - - let string = ''; - let name; - let shouldShow = false; - if (appInfo.get_display_name) { - // show only launchers that should be visible in this DE - shouldShow = appInfo.should_show() && this._parentalControlsManager.shouldShowApp(appInfo); - - if (shouldShow) { - let id = appInfo.get_id().split('.'); - id = id[id.length - 2] || ''; - let baseName = appInfo.get_string('Name') || ''; - let dispName = appInfo.get_display_name() || ''; - let gName = appInfo.get_generic_name() || ''; - let description = appInfo.get_description() || ''; - let categories = appInfo.get_string('Categories')?.replace(/;/g, ' ') || ''; - let keywords = appInfo.get_string('Keywords')?.replace(/;/g, ' ') || ''; - name = `${dispName} ${id}`; - string = `${dispName} ${gName} ${baseName} ${description} ${categories} ${keywords} ${id}`; - } - } - - let m = -1; - if (shouldShow && opt.SEARCH_FUZZY) { - m = Me.Util.fuzzyMatch(pattern, name); - m = (m + Me.Util.strictMatch(pattern, string)) / 2; - } else if (shouldShow) { - m = Me.Util.strictMatch(pattern, string); - } - - if (m !== -1) - weightList[appInfo.get_id()] = m; - - return shouldShow && (m !== -1); - }); - - appInfoList.sort((a, b) => weightList[a.get_id()] > weightList[b.get_id()]); - - const usage = Shell.AppUsage.get_default(); - // sort apps by usage list - appInfoList.sort((a, b) => usage.compare(a.get_id(), b.get_id())); - // prefer apps where any word in their name starts with the pattern - appInfoList.sort((a, b) => Me.Util.isMoreRelevant(a.get_display_name(), b.get_display_name(), pattern)); - - let results = appInfoList.map(app => app.get_id()); - - if (opt.SEARCH_APP_GRID_MODE && Main.overview.dash.showAppsButton.checked) - this._filterAppGrid(results); - - results = results.concat(this._systemActions.getMatchingActions(terms)); - - return new Promise(resolve => resolve(results)); - }, - - _filterAppGrid(results) { - const icons = Main.overview._overview.controls._appDisplay._orderedItems; - icons.forEach(icon => { - icon.visible = results.includes(icon.id); - }); - }, - - // App search result size - createResultObject(resultMeta) { - let iconSize = opt.SEARCH_ICON_SIZE; - if (!iconSize) { - iconSize = Me.Util.monitorHasLowResolution() - ? 64 - : 96; - } - - if (resultMeta.id.endsWith('.desktop')) { - const icon = new AppDisplay.AppIcon(this._appSys.lookup_app(resultMeta['id']), { - expandTitleOnHover: false, - }); - icon.icon.setIconSize(iconSize); - return icon; - } else { - this._iconSize = iconSize; - return new SystemActionIcon(this, resultMeta); - } - }, -}; - -const SystemActionIcon = GObject.registerClass({ - // Registered name should be unique - GTypeName: `SystemAction${Math.floor(Math.random() * 1000)}`, -}, class SystemActionIcon extends Search.GridSearchResult { - _init(provider, metaInfo, resultsView) { - super._init(provider, metaInfo, resultsView); - if (!Clutter.Container) - this.add_style_class_name('grid-search-result-46'); - this.icon._setSizeManually = true; - this.icon.setIconSize(provider._iconSize); - } - - activate() { - SystemActions.getDefault().activateAction(this.metaInfo['id']); - Main.overview.hide(); - } -}); - -const SearchResult = { - activate() { - this.provider.activateResult(this.metaInfo.id, this._resultsView.terms); - - if (this.metaInfo.clipboardText) { - St.Clipboard.get_default().set_text( - St.ClipboardType.CLIPBOARD, this.metaInfo.clipboardText); - } - // don't close overview if Shift key is pressed - Shift moves windows to the workspace - if (!Me.Util.isShiftPressed()) - Main.overview.toggle(); - }, -}; - -const SearchResultsView = { - setTerms(terms) { - // Check for the case of making a duplicate previous search before - // setting state of the current search or cancelling the search. - // This will prevent incorrect state being as a result of a duplicate - // search while the previous search is still active. - let searchString = terms.join(' '); - let previousSearchString = this._terms.join(' '); - if (searchString === previousSearchString) - return; - - this._startingSearch = true; - - this._cancellable.cancel(); - this._cancellable.reset(); - - if (terms.length === 0) { - this._reset(); - return; - } - - let isSubSearch = false; - if (this._terms.length > 0) - isSubSearch = searchString.indexOf(previousSearchString) === 0; - - this._terms = terms; - this._isSubSearch = isSubSearch; - this._updateSearchProgress(); - - if (!this._searchTimeoutId) - this._searchTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, opt.SEARCH_DELAY, this._onSearchTimeout.bind(this)); - - this._highlighter = new Highlighter(this._terms); - - this.emit('terms-changed'); - }, - - _doSearch() { - this._startingSearch = false; - - let previousResults = this._results; - this._results = {}; - - const term0 = this._terms[0]; - const onlySupportedProviders = term0.startsWith(Me.WSP_PREFIX) || term0.startsWith(Me.ESP_PREFIX) || term0.startsWith(Me.RFSP_PREFIX); - - this._providers.forEach(provider => { - const supportedProvider = ['open-windows', 'extensions', 'recent-files'].includes(provider.id); - if (!onlySupportedProviders || (onlySupportedProviders && supportedProvider)) { - let previousProviderResults = previousResults[provider.id]; - this._doProviderSearch(provider, previousProviderResults); - } else { - // hide unwanted providers, they will show() automatically when needed - provider.display.visible = false; - } - }); - - this._updateSearchProgress(); - this._clearSearchTimeout(); - }, - - _updateSearchProgress() { - let haveResults = this._providers.some(provider => { - let display = provider.display; - return display.getFirstResult() !== null; - }); - - this._scrollView.visible = haveResults; - this._statusBin.visible = !haveResults; - - if (!haveResults) { - if (this.searchInProgress) - this._statusText.set_text(_('Searching…')); - else - this._statusText.set_text(_('No results.')); - } - }, - - _highlightFirstVisibleAppGridIcon() { - const appDisplay = Main.overview._overview.controls._appDisplay; - // appDisplay.grab_key_focus(); - for (const icon of appDisplay._orderedItems) { - if (icon.visible) { - appDisplay.selectApp(icon.id); - break; - } - } - }, - - _maybeSetInitialSelection() { - if (opt.SEARCH_APP_GRID_MODE && Main.overview.dash.showAppsButton.checked) { - this._highlightFirstVisibleAppGridIcon(); - return; - } - - let newDefaultResult = null; - - let providers = this._providers; - for (let i = 0; i < providers.length; i++) { - let provider = providers[i]; - let display = provider.display; - - if (!display.visible) - continue; - - let firstResult = display.getFirstResult(); - if (firstResult) { - newDefaultResult = firstResult; - break; // select this one! - } - } - - if (newDefaultResult !== this._defaultResult) { - this._setSelected(this._defaultResult, false); - this._setSelected(newDefaultResult, this._highlightDefault); - - this._defaultResult = newDefaultResult; - } - }, - - highlightDefault(highlight) { - if (opt.SEARCH_APP_GRID_MODE && Main.overview.dash.showAppsButton.checked) { - if (highlight) - this._highlightFirstVisibleAppGridIcon(); - } else { - this._highlightDefault = highlight; - this._setSelected(this._defaultResult, highlight); - } - }, -}; - -// Add highlighting of the "name" part of the result for all providers -const ListSearchResultOverride = { - _highlightTerms() { - let markup = this._resultsView.highlightTerms(this.metaInfo['name']); - this.label_actor.clutter_text.set_markup(markup); - markup = this._resultsView.highlightTerms(this.metaInfo['description'].split('\n')[0]); - this._descriptionLabel.clutter_text.set_markup(markup); - }, -}; - -const HighlighterOverride = { - /** - * @param {?string[]} terms - list of terms to highlight - */ - /* constructor(terms) { - if (!terms) - return; - - const escapedTerms = terms - .map(term => Shell.util_regex_escape(term)) - .filter(term => term.length > 0); - - if (escapedTerms.length === 0) - return; - - this._highlightRegex = new RegExp( - `(${escapedTerms.join('|')})`, 'gi'); - },*/ - - /** - * Highlight all occurences of the terms defined for this - * highlighter in the provided text using markup. - * - * @param {string} text - text to highlight the defined terms in - * @returns {string} - */ - highlight(text, options) { - if (!this._highlightRegex) - return GLib.markup_escape_text(text, -1); - - // force use local settings if the class is overridden by another extension (WSP, ESP) - const o = options || opt; - let escaped = []; - let lastMatchEnd = 0; - let match; - let style = ['', '']; - if (o.HIGHLIGHT_DEFAULT) - style = ['<b>', '</b>']; - // The default highlighting by the bold style causes text to be "randomly" ellipsized in cases where it's not necessary - // and also blurry - // Underscore doesn't affect label size and all looks better - else if (o.HIGHLIGHT_UNDERLINE) - style = ['<u>', '</u>']; - - while ((match = this._highlightRegex.exec(text))) { - if (match.index > lastMatchEnd) { - let unmatched = GLib.markup_escape_text( - text.slice(lastMatchEnd, match.index), -1); - escaped.push(unmatched); - } - let matched = GLib.markup_escape_text(match[0], -1); - escaped.push(`${style[0]}${matched}${style[1]}`); - lastMatchEnd = match.index + match[0].length; - } - let unmatched = GLib.markup_escape_text( - text.slice(lastMatchEnd), -1); - escaped.push(unmatched); - return escaped.join(''); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/searchController.js b/extensions/46/vertical-workspaces/lib/searchController.js deleted file mode 100644 index 1722f15..0000000 --- a/extensions/46/vertical-workspaces/lib/searchController.js +++ /dev/null @@ -1,94 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * searchController.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; - -let Me; -let opt; - -export const SearchControllerModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._originalOnStageKeyPress = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('searchControllerModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' SearchControllerModule - Keeping untouched'); - } - - _activateModule() { - if (!this._originalOnStageKeyPress) - this._originalOnStageKeyPress = Main.overview.searchController._onStageKeyPress; - - Main.overview.searchController._onStageKeyPress = SearchControllerCommon._onStageKeyPress; - console.debug(' SearchControllerModule - Activated'); - } - - _disableModule() { - if (this._originalOnStageKeyPress) - Main.overview.searchController._onStageKeyPress = this._originalOnStageKeyPress; - this._originalOnStageKeyPress = null; - - console.debug(' SearchControlerModule - Disabled'); - } -}; - -// if opt.ESC_BEHAVIOR > 0 force close the overview -const SearchControllerCommon = { - _onStageKeyPress(actor, event) { - // Ignore events while anything but the overview has - // pushed a modal (system modals, looking glass, ...) - if (Main.modalCount > 1) - return Clutter.EVENT_PROPAGATE; - - let symbol = event.get_key_symbol(); - if (symbol === Clutter.KEY_Escape) { - if (this._searchActive && !opt.ESC_BEHAVIOR) { - this.reset(); - } else if (this._showAppsButton.checked && !opt.ESC_BEHAVIOR) { - this._showAppsButton.checked = false; - } else { - this.reset(); - Main.overview.hide(); - } - - return Clutter.EVENT_STOP; - } else if (this._shouldTriggerSearch(symbol)) { - this.startSearch(event); - } - return Clutter.EVENT_PROPAGATE; - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/settings.js b/extensions/46/vertical-workspaces/lib/settings.js deleted file mode 100644 index 563063c..0000000 --- a/extensions/46/vertical-workspaces/lib/settings.js +++ /dev/null @@ -1,539 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * settings.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - */ - -'use strict'; - -import GLib from 'gi://GLib'; - -let Me; - -export const Options = class Options { - constructor(me) { - Me = me; - - this._gsettings = Me.gSettings; - this._connectionIds = []; - this._writeTimeoutId = 0; - this._gsettings.delay(); - this.connect('changed', () => { - if (this._writeTimeoutId) - GLib.Source.remove(this._writeTimeoutId); - - this._writeTimeoutId = GLib.timeout_add( - GLib.PRIORITY_DEFAULT, - 400, - () => { - this._gsettings.apply(); - this._updateSettings(); - this._writeTimeoutId = 0; - return GLib.SOURCE_REMOVE; - } - ); - }); - this.options = { - workspaceThumbnailsPosition: ['int', 'ws-thumbnails-position'], - wsMaxSpacing: ['int', 'ws-max-spacing'], - wsPreviewScale: ['int', 'ws-preview-scale'], - secWsPreviewScale: ['int', 'secondary-ws-preview-scale'], - secWsPreviewShift: ['boolean', 'secondary-ws-preview-shift'], - wsThumbnailsFull: ['boolean', 'ws-thumbnails-full'], - secWsThumbnailsPosition: ['int', 'secondary-ws-thumbnails-position'], - dashPosition: ['int', 'dash-position'], - dashPositionAdjust: ['int', 'dash-position-adjust'], - wsTmbPositionAdjust: ['int', 'wst-position-adjust'], - showWsTmbLabels: ['int', 'show-wst-labels'], - showWsTmbLabelsOnHover: ['boolean', 'show-wst-labels-on-hover'], - closeWsButtonMode: ['int', 'close-ws-button-mode'], - secWsTmbPositionAdjust: ['int', 'sec-wst-position-adjust'], - dashMaxIconSize: ['int', 'dash-max-icon-size'], - centerDashToWs: ['boolean', 'center-dash-to-ws'], - showAppsIconPosition: ['int', 'show-app-icon-position'], - wsThumbnailScale: ['int', 'ws-thumbnail-scale'], - wsThumbnailScaleAppGrid: ['int', 'ws-thumbnail-scale-appgrid'], - secWsThumbnailScale: ['int', 'secondary-ws-thumbnail-scale'], - showSearchEntry: ['boolean', 'show-search-entry'], - centerSearch: ['boolean', 'center-search'], - centerAppGrid: ['boolean', 'center-app-grid'], - dashBgOpacity: ['int', 'dash-bg-opacity'], - dashBgColor: ['int', 'dash-bg-color'], - dashBgRadius: ['int', 'dash-bg-radius'], - dashBgGS3Style: ['boolean', 'dash-bg-gs3-style'], - runningDotStyle: ['int', 'running-dot-style'], - enablePageShortcuts: ['boolean', 'enable-page-shortcuts'], - showWsSwitcherBg: ['boolean', 'show-ws-switcher-bg'], - showWsPreviewBg: ['boolean', 'show-ws-preview-bg'], - wsPreviewBgRadius: ['int', 'ws-preview-bg-radius'], - showBgInOverview: ['boolean', 'show-bg-in-overview'], - overviewBgBrightness: ['int', 'overview-bg-brightness'], - searchBgBrightness: ['int', 'search-bg-brightness'], - overviewBgBlurSigma: ['int', 'overview-bg-blur-sigma'], - appGridBgBlurSigma: ['int', 'app-grid-bg-blur-sigma'], - smoothBlurTransitions: ['boolean', 'smooth-blur-transitions'], - appGridAnimation: ['int', 'app-grid-animation'], - searchViewAnimation: ['int', 'search-view-animation'], - workspaceAnimation: ['int', 'workspace-animation'], - animationSpeedFactor: ['int', 'animation-speed-factor'], - winPreviewIconSize: ['int', 'win-preview-icon-size'], - winTitlePosition: ['int', 'win-title-position'], - startupState: ['int', 'startup-state'], - overviewMode: ['int', 'overview-mode'], - workspaceSwitcherAnimation: ['int', 'workspace-switcher-animation'], - wsSwitcherMode: ['int', 'ws-switcher-mode'], - searchIconSize: ['int', 'search-icon-size'], - searchViewScale: ['int', 'search-width-scale'], - appGridIconSize: ['int', 'app-grid-icon-size'], - appGridColumns: ['int', 'app-grid-columns'], - appGridRows: ['int', 'app-grid-rows'], - appGridFolderIconSize: ['int', 'app-grid-folder-icon-size'], - appGridFolderColumns: ['int', 'app-grid-folder-columns'], - appGridFolderRows: ['int', 'app-grid-folder-rows'], - appGridFolderIconGrid: ['int', 'app-grid-folder-icon-grid'], - appGridContent: ['int', 'app-grid-content'], - appGridIncompletePages: ['boolean', 'app-grid-incomplete-pages'], - appGridOrder: ['int', 'app-grid-order'], - appFolderOrder: ['int', 'app-folder-order'], - appGridNamesMode: ['int', 'app-grid-names'], - appGridActivePreview: ['boolean', 'app-grid-active-preview'], - appGridFolderCenter: ['boolean', 'app-grid-folder-center'], - appGridPageWidthScale: ['int', 'app-grid-page-width-scale'], - appGridPageHeightScale: ['int', 'app-grid-page-height-scale'], - appGridSpacing: ['int', 'app-grid-spacing'], - appGridFolderSpacing: ['int', 'app-grid-folder-spacing'], - appGridShowPageArrows: ['boolean', 'app-grid-show-page-arrows'], - searchWindowsOrder: ['int', 'search-windows-order'], - searchFuzzy: ['boolean', 'search-fuzzy'], - searchMaxResultsRows: ['int', 'search-max-results-rows'], - searchAppGridMode: ['int', 'search-app-grid-mode'], - dashShowWindowsBeforeActivation: ['int', 'dash-show-windows-before-activation'], - dashIconScroll: ['int', 'dash-icon-scroll'], - dashIsolateWorkspaces: ['boolean', 'dash-isolate-workspaces'], - searchWindowsIconScroll: ['int', 'search-windows-icon-scroll'], - panelVisibility: ['int', 'panel-visibility'], - panelPosition: ['int', 'panel-position'], - windowAttentionMode: ['int', 'window-attention-mode'], - wsSwPopupHPosition: ['int', 'ws-sw-popup-h-position'], - wsSwPopupVPosition: ['int', 'ws-sw-popup-v-position'], - wsSwPopupMode: ['int', 'ws-sw-popup-mode'], - wsSwitcherWraparound: ['boolean', 'ws-switcher-wraparound'], - wsSwitcherIgnoreLast: ['boolean', 'ws-switcher-ignore-last'], - favoritesNotify: ['int', 'favorites-notify'], - notificationPosition: ['int', 'notification-position'], - osdPosition: ['int', 'osd-position'], - hotCornerAction: ['int', 'hot-corner-action'], - hotCornerPosition: ['int', 'hot-corner-position'], - hotCornerFullscreen: ['boolean', 'hot-corner-fullscreen'], - hotCornerRipples: ['boolean', 'hot-corner-ripples'], - alwaysActivateSelectedWindow: ['boolean', 'always-activate-selected-window'], - winPreviewSecBtnAction: ['int', 'win-preview-sec-mouse-btn-action'], - winPreviewMidBtnAction: ['int', 'win-preview-mid-mouse-btn-action'], - winPreviewShowCloseButton: ['boolean', 'win-preview-show-close-button'], - windowIconClickAction: ['int', 'window-icon-click-action'], - overlayKeyPrimary: ['int', 'overlay-key-primary'], - overlayKeySecondary: ['int', 'overlay-key-secondary'], - overviewEscBehavior: ['int', 'overview-esc-behavior'], - clickEmptyClose: ['boolean', 'click-empty-close'], - newWindowFocusFix: ['boolean', 'new-window-focus-fix'], - newWindowMonitorFix: ['boolean', 'new-window-monitor-fix'], - appGridPerformance: ['boolean', 'app-grid-performance'], - highlightingStyle: ['int', 'highlighting-style'], - delayStartup: ['boolean', 'delay-startup'], - - workspaceSwitcherPopupModule: ['boolean', 'workspace-switcher-popup-module'], - workspaceAnimationModule: ['boolean', 'workspace-animation-module'], - workspaceModule: ['boolean', 'workspace-module'], - windowManagerModule: ['boolean', 'window-manager-module'], - windowPreviewModule: ['boolean', 'window-preview-module'], - windowAttentionHandlerModule: ['boolean', 'win-attention-handler-module'], - swipeTrackerModule: ['boolean', 'swipe-tracker-module'], - searchControllerModule: ['boolean', 'search-controller-module'], - searchModule: ['boolean', 'search-module'], - panelModule: ['boolean', 'panel-module'], - overlayKeyModule: ['boolean', 'overlay-key-module'], - osdWindowModule: ['boolean', 'osd-window-module'], - messageTrayModule: ['boolean', 'message-tray-module'], - layoutModule: ['boolean', 'layout-module'], - dashModule: ['boolean', 'dash-module'], - appFavoritesModule: ['boolean', 'app-favorites-module'], - appDisplayModule: ['boolean', 'app-display-module'], - - profileName1: ['string', 'profile-name-1'], - profileName2: ['string', 'profile-name-2'], - profileName3: ['string', 'profile-name-3'], - profileName4: ['string', 'profile-name-4'], - }; - this.cachedOptions = {}; - this._updateSettings(); - } - - connect(name, callback) { - const id = this._gsettings.connect(name, callback); - this._connectionIds.push(id); - return id; - } - - destroy() { - this._connectionIds.forEach(id => this._gsettings.disconnect(id)); - if (this._writeTimeoutId) { - GLib.source_remove(this._writeTimeoutId); - this._writeTimeoutId = 0; - } - - Me = null; - } - - _updateCachedSettings() { - Object.keys(this.options).forEach(v => this.get(v, true)); - } - - get(option, updateCache = false) { - if (!this.options[option]) { - console.error(`[${Me.metadata.name}] Error: Option ${option} is undefined.`); - return null; - } - - if (updateCache || this.cachedOptions[option] === undefined) { - const [, key, settings] = this.options[option]; - let gSettings; - if (settings !== undefined) - gSettings = settings(); - else - gSettings = this._gsettings; - - this.cachedOptions[option] = gSettings.get_value(key).deep_unpack(); - } - - return this.cachedOptions[option]; - } - - set(option, value) { - const [format, key, settings] = this.options[option]; - - let gSettings = this._gsettings; - - if (settings !== undefined) - gSettings = settings(); - - - switch (format) { - case 'boolean': - gSettings.set_boolean(key, value); - break; - case 'int': - gSettings.set_int(key, value); - break; - case 'string': - gSettings.set_string(key, value); - break; - case 'strv': - gSettings.set_strv(key, value); - break; - } - } - - getDefault(option) { - const [, key, settings] = this.options[option]; - - let gSettings = this._gsettings; - - if (settings !== undefined) - gSettings = settings(); - - - return gSettings.get_default_value(key).deep_unpack(); - } - - storeProfile(index) { - const profile = {}; - Object.keys(this.options).forEach(v => { - if (!v.startsWith('profileName')) - profile[v] = this.get(v).toString(); - }); - - this._gsettings.set_value(`profile-data-${index}`, new GLib.Variant('a{ss}', profile)); - } - - loadProfile(index) { - const options = this._gsettings.get_value(`profile-data-${index}`).deep_unpack(); - // set the aaa-loading-data so extension.js doesn't reset V-Shell after each profile item - // delayed gsettings writes are processed alphabetically, so this key will be processed first - this._gsettings.set_boolean('aaa-loading-profile', !this._gsettings.get_boolean('aaa-loading-profile')); - for (let o of Object.keys(options)) { - if (!this.options[o]) { - console.error(`[${Me.metadata.name}] Error: "${o}" is not a valid profile key -> Update your profile`); - continue; - } - const [type] = this.options[o]; - let value = options[o]; - switch (type) { - case 'string': - break; - case 'boolean': - value = value === 'true'; - break; - case 'int': - value = parseInt(value); - break; - } - - this.set(o, value); - } - } - - resetProfile(index) { - this._gsettings.reset(`profile-data-${index}`); - this._gsettings.reset(`profile-name-${index}`); - } - - _updateSettings() { - this._updateCachedSettings(); - - // Basic spacing of the overview elements - this.SPACING = 12; - - this.DASH_BG_ALPHA = this.get('dashBgOpacity') / 100; - this.DASH_BG_OPACITY = this.get('dashBgOpacity') * 2.5; - this.DASH_BG_COLOR = this.get('dashBgColor'); - this.DASH_BG_RADIUS = this.get('dashBgRadius'); - this.DASH_BG_LIGHT = this.DASH_BG_COLOR === 1; - this.DASH_BG_GS3_STYLE = this.get('dashBgGS3Style'); - this.DASH_POSITION = this.get('dashModule') ? this.get('dashPosition') : 2; - this.DASH_TOP = this.DASH_POSITION === 0; - this.DASH_RIGHT = this.DASH_POSITION === 1; - this.DASH_BOTTOM = this.DASH_POSITION === 2; - this.DASH_LEFT = this.DASH_POSITION === 3; - this.DASH_VERTICAL = this.DASH_LEFT || this.DASH_RIGHT; - this.DASH_VISIBLE = this.DASH_POSITION !== 4; // 4 - disable - this.DASH_FOLLOW_RECENT_WIN = false; - - this.DASH_ISOLATE_WS = this.get('dashIsolateWorkspaces'); - - this.DASH_CLICK_ACTION = this.get('dashShowWindowsBeforeActivation'); - this.DASH_CLICK_SWITCH_BEFORE_ACTIVATION = this.DASH_CLICK_ACTION === 1; - this.DASH_CLICK_OPEN_NEW_WIN = this.DASH_CLICK_ACTION === 2; - this.DASH_CLICK_PREFER_WORKSPACE = this.DASH_CLICK_ACTION === 3; - - this.DASH_ICON_SCROLL = this.get('dashIconScroll'); - this.DASH_SHIFT_CLICK_MV = true; - - this.RUNNING_DOT_STYLE = this.get('runningDotStyle'); - - this.SEARCH_WINDOWS_ICON_SCROLL = this.get('searchWindowsIconScroll'); - - this.DASH_POSITION_ADJUSTMENT = this.get('dashPositionAdjust'); - this.DASH_POSITION_ADJUSTMENT = this.DASH_POSITION_ADJUSTMENT * -1 / 100; // range 1 to -1 - this.CENTER_DASH_WS = this.get('centerDashToWs'); - - this.MAX_ICON_SIZE = this.get('dashMaxIconSize'); - - this.WS_TMB_POSITION = this.get('workspaceThumbnailsPosition'); - this.ORIENTATION = this.WS_TMB_POSITION > 4 ? 0 : 1; - this.WORKSPACE_MAX_SPACING = this.get('wsMaxSpacing'); - this.WS_MAX_SPACING_OFF_SCREEN = 350; - this.FORCE_SINGLE_WS_TRANSITION = false; - // ORIENTATION || DASH_LEFT || DASH_RIGHT ? 350 : 80; - this.SHOW_WS_TMB = ![4, 9].includes(this.WS_TMB_POSITION); // 4, 9 - disable - this.WS_TMB_FULL = this.get('wsThumbnailsFull'); - // translate ws tmb position to 0 top, 1 right, 2 bottom, 3 left - // 0L 1R, 2LF, 3RF, 4DV, 5T, 6B, 7TF, 8BF, 9DH - this.WS_TMB_POSITION = [3, 1, 3, 1, 4, 0, 2, 0, 2, 8][this.WS_TMB_POSITION]; - this.WS_TMB_TOP = this.WS_TMB_POSITION === 0; - this.WS_TMB_RIGHT = this.WS_TMB_POSITION === 1; - this.WS_TMB_BOTTOM = this.WS_TMB_POSITION === 2; - this.WS_TMB_LEFT = this.WS_TMB_POSITION === 3; - this.WS_TMB_POSITION_ADJUSTMENT = this.get('wsTmbPositionAdjust') * -1 / 100; // range 1 to -1 - this.SEC_WS_TMB_POSITION = this.get('secWsThumbnailsPosition'); - this.SHOW_SEC_WS_TMB = this.SEC_WS_TMB_POSITION !== 3 && this.SHOW_WS_TMB; - this.SEC_WS_TMB_TOP = (this.SEC_WS_TMB_POSITION === 0 && !this.ORIENTATION) || (this.SEC_WS_TMB_POSITION === 2 && this.WS_TMB_TOP); - this.SEC_WS_TMB_RIGHT = (this.SEC_WS_TMB_POSITION === 1 && this.ORIENTATION) || (this.SEC_WS_TMB_POSITION === 2 && this.WS_TMB_RIGHT); - this.SEC_WS_TMB_BOTTOM = (this.SEC_WS_TMB_POSITION === 1 && !this.ORIENTATION) || (this.SEC_WS_TMB_POSITION === 2 && this.WS_TMB_BOTTOM); - this.SEC_WS_TMB_LEFT = (this.SEC_WS_TMB_POSITION === 0 && this.ORIENTATION) || (this.SEC_WS_TMB_POSITION === 2 && this.WS_TMB_LEFT); - - this.SEC_WS_TMB_POSITION_ADJUSTMENT = this.get('secWsTmbPositionAdjust') * -1 / 100; // range 1 to -1 - this.SEC_WS_PREVIEW_SHIFT = this.get('secWsPreviewShift'); - this.SHOW_WST_LABELS = this.get('showWsTmbLabels'); - this.SHOW_WST_LABELS_ON_HOVER = this.get('showWsTmbLabelsOnHover'); - this.CLOSE_WS_BUTTON_MODE = this.get('closeWsButtonMode'); - - this.MAX_THUMBNAIL_SCALE = this.get('wsThumbnailScale') / 100 + 0.01; - this.MAX_THUMBNAIL_SCALE_APPGRID = this.get('wsThumbnailScaleAppGrid') / 100 + 0.01; - this.SHOW_WS_TMB_APPGRID = true; - this.MAX_THUMBNAIL_SCALE_STABLE = this.MAX_THUMBNAIL_SCALE === this.MAX_THUMBNAIL_SCALE_APPGRID; - this.SEC_MAX_THUMBNAIL_SCALE = this.get('secWsThumbnailScale') / 100 + 0.01; - - this.WS_PREVIEW_SCALE = this.get('wsPreviewScale') / 100; - this.SEC_WS_PREVIEW_SCALE = this.get('secWsPreviewScale') / 100; - // calculate number of possibly visible neighbor previews according to ws scale - this.NUMBER_OF_VISIBLE_NEIGHBORS = Math.round(2 + (1 - this.WS_PREVIEW_SCALE)); - - this.SHOW_WS_TMB_BG = this.get('showWsSwitcherBg') && this.SHOW_WS_TMB; - this.WS_PREVIEW_BG_RADIUS = this.get('wsPreviewBgRadius'); - this.SHOW_WS_PREVIEW_BG = this.get('showWsPreviewBg'); - - this.CENTER_APP_GRID = this.get('centerAppGrid'); - - this.SHOW_SEARCH_ENTRY = this.get('showSearchEntry'); - this.CENTER_SEARCH_VIEW = this.get('centerSearch'); - this.APP_GRID_ANIMATION = this.get('appGridAnimation'); - if (this.APP_GRID_ANIMATION === 4) - this.APP_GRID_ANIMATION = this._getAnimationDirection(); - - this.SEARCH_VIEW_ANIMATION = this.get('searchViewAnimation'); - if (this.SEARCH_VIEW_ANIMATION === 4) - this.SEARCH_VIEW_ANIMATION = 3; - - this.WIN_PREVIEW_ICON_SIZE = [64, 48, 32, 22, 8][this.get('winPreviewIconSize')]; - this.WIN_TITLES_POSITION = this.get('winTitlePosition'); - this.ALWAYS_SHOW_WIN_TITLES = this.WIN_TITLES_POSITION === 1; - - this.STARTUP_STATE = this.get('startupState'); - this.SHOW_BG_IN_OVERVIEW = this.get('showBgInOverview'); - this.OVERVIEW_BG_BRIGHTNESS = this.get('overviewBgBrightness') / 100; - this.SEARCH_BG_BRIGHTNESS = this.get('searchBgBrightness') / 100; - this.OVERVIEW_BG_BLUR_SIGMA = this.get('overviewBgBlurSigma'); - this.APP_GRID_BG_BLUR_SIGMA = this.get('appGridBgBlurSigma'); - this.SMOOTH_BLUR_TRANSITIONS = this.get('smoothBlurTransitions'); - - this.OVERVIEW_MODE = this.get('overviewMode'); - this.OVERVIEW_MODE2 = this.OVERVIEW_MODE === 2; - this.WORKSPACE_MODE = this.OVERVIEW_MODE ? 0 : 1; - - this.STATIC_WS_SWITCHER_BG = this.get('workspaceSwitcherAnimation'); - - this.ANIMATION_TIME_FACTOR = this.get('animationSpeedFactor') / 100; - - this.SEARCH_ICON_SIZE = this.get('searchIconSize'); - this.SEARCH_VIEW_SCALE = this.get('searchViewScale') / 100; - this.SEARCH_MAX_ROWS = this.get('searchMaxResultsRows'); - this.SEARCH_FUZZY = this.get('searchFuzzy'); - this.SEARCH_DELAY = 0; - this.SEARCH_APP_GRID_MODE = this.get('searchAppGridMode'); - - this.APP_GRID_ALLOW_INCOMPLETE_PAGES = this.get('appGridIncompletePages'); - this.APP_GRID_ICON_SIZE = this.get('appGridIconSize'); - this.APP_GRID_COLUMNS = this.get('appGridColumns'); - this.APP_GRID_ROWS = this.get('appGridRows'); - this.APP_GRID_ADAPTIVE = !this.APP_GRID_COLUMNS && !this.APP_GRID_ROWS; - - this.APP_GRID_ORDER = this.get('appGridOrder'); - this.APP_GRID_ALPHABET = [1, 2, 4].includes(this.APP_GRID_ORDER); - this.APP_GRID_FOLDERS_FIRST = this.APP_GRID_ORDER === 1; - this.APP_GRID_FOLDERS_LAST = this.APP_GRID_ORDER === 2; - this.APP_GRID_USAGE = this.APP_GRID_ORDER === 3; - - this.APP_FOLDER_ORDER = this.get('appFolderOrder'); - this.APP_FOLDER_ALPHABET = this.APP_FOLDER_ORDER === 1; - this.APP_FOLDER_USAGE = this.APP_FOLDER_ORDER === 2; - - this.APP_GRID_INCLUDE_DASH = this.get('appGridContent'); - /* APP_GRID_INCLUDE_DASH - 0 - Include All - 1 - Include All - Favorites and Runnings First - 2 - Exclude Favorites (Default) - 3 - Exclude Running - 4 - Exclude Favorites and Running - */ - this.APP_GRID_EXCLUDE_FAVORITES = this.APP_GRID_INCLUDE_DASH === 2 || this.APP_GRID_INCLUDE_DASH === 4; - this.APP_GRID_EXCLUDE_RUNNING = this.APP_GRID_INCLUDE_DASH === 3 || this.APP_GRID_INCLUDE_DASH === 4; - this.APP_GRID_DASH_FIRST = this.APP_GRID_INCLUDE_DASH === 1; - - this.APP_GRID_NAMES_MODE = this.get('appGridNamesMode'); - - this.APP_GRID_FOLDER_ICON_SIZE = this.get('appGridFolderIconSize'); - this.APP_GRID_FOLDER_ICON_GRID = this.get('appGridFolderIconGrid'); - this.APP_GRID_FOLDER_COLUMNS = this.get('appGridFolderColumns'); - this.APP_GRID_FOLDER_ROWS = this.get('appGridFolderRows'); - this.APP_GRID_SPACING = this.get('appGridSpacing'); - this.APP_GRID_FOLDER_SPACING = this.get('appGridFolderSpacing'); - this.APP_GRID_FOLDER_DEFAULT = this.APP_GRID_FOLDER_ROWS === 3 && this.APP_GRID_FOLDER_COLUMNS === 3; - this.APP_GRID_FOLDER_ADAPTIVE = !this.APP_GRID_FOLDER_COLUMNS && !this.APP_GRID_FOLDER_ROWS; - this.APP_GRID_ACTIVE_PREVIEW = this.get('appGridActivePreview'); - this.APP_GRID_FOLDER_CENTER = this.get('appGridFolderCenter'); - this.APP_GRID_PAGE_WIDTH_SCALE = this.get('appGridPageWidthScale') / 100; - this.APP_GRID_PAGE_HEIGHT_SCALE = this.get('appGridPageHeightScale') / 100; - this.APP_GRID_SHOW_PAGE_ARROWS = this.get('appGridShowPageArrows'); - - // Default icon sizes updates in the IconGrid._findBestModeForSize() - this.APP_GRID_ICON_SIZE_DEFAULT = this.APP_GRID_ACTIVE_PREVIEW && !this.APP_GRID_USAGE ? 192 : 96; - this.APP_GRID_FOLDER_ICON_SIZE_DEFAULT = 96; - - this.APP_GRID_PERFORMANCE = this.get('appGridPerformance'); - - this.PANEL_POSITION_TOP = this.get('panelPosition') === 0; - this.PANEL_POSITION_BOTTOM = this.get('panelPosition') === 1; - this.PANEL_MODE = this.get('panelVisibility'); - this.PANEL_DISABLED = this.PANEL_MODE === 2; - this.PANEL_OVERVIEW_ONLY = this.PANEL_MODE === 1; - - this.WINDOW_ATTENTION_MODE = this.get('windowAttentionMode'); - this.WINDOW_ATTENTION_DISABLE_NOTIFICATIONS = this.WINDOW_ATTENTION_MODE === 1; - this.WINDOW_ATTENTION_FOCUS_IMMEDIATELY = this.WINDOW_ATTENTION_MODE === 2; - - this.WS_SW_POPUP_H_POSITION = this.get('wsSwPopupHPosition') / 100; - this.WS_SW_POPUP_V_POSITION = this.get('wsSwPopupVPosition') / 100; - this.WS_SW_POPUP_MODE = this.get('wsSwPopupMode'); - - this.WS_ANIMATION = this.get('workspaceAnimation'); - this.WS_WRAPAROUND = this.get('wsSwitcherWraparound'); - this.WS_IGNORE_LAST = this.get('wsSwitcherIgnoreLast'); - this.WS_SWITCHER_CURRENT_MONITOR = this.get('wsSwitcherMode') === 1; - - this.SHOW_FAV_NOTIFICATION = this.get('favoritesNotify'); - this.NOTIFICATION_POSITION = this.get('notificationPosition'); - - this.OSD_POSITION = this.get('osdPosition'); - - this.HOT_CORNER_ACTION = this.get('hotCornerAction'); - this.HOT_CORNER_POSITION = this.get('hotCornerPosition'); - if (this.HOT_CORNER_POSITION === 6 && this.DASH_VISIBLE) - this.HOT_CORNER_EDGE = true; - else - this.HOT_CORNER_EDGE = false; - if ([5, 6].includes(this.HOT_CORNER_POSITION)) { - if (this.DASH_TOP || this.DASH_LEFT) - this.HOT_CORNER_POSITION = 1; - else if (this.DASH_RIGHT) - this.HOT_CORNER_POSITION = 2; - else if (this.DASH_BOTTOM) - this.HOT_CORNER_POSITION = 3; - else - this.HOT_CORNER_POSITION = 0; - } - this.HOT_CORNER_FULLSCREEN = this.get('hotCornerFullscreen'); - this.HOT_CORNER_RIPPLES = this.get('hotCornerRipples'); - - this.ALWAYS_ACTIVATE_SELECTED_WINDOW = this.get('alwaysActivateSelectedWindow'); - this.WIN_PREVIEW_SEC_BTN_ACTION = this.get('winPreviewSecBtnAction'); - this.WIN_PREVIEW_MID_BTN_ACTION = this.get('winPreviewMidBtnAction'); - this.SHOW_CLOSE_BUTTON = this.get('winPreviewShowCloseButton'); - this.WINDOW_ICON_CLICK_ACTION = this.get('windowIconClickAction'); - - this.OVERLAY_KEY_PRIMARY = this.get('overlayKeyPrimary'); - this.OVERLAY_KEY_SECONDARY = this.get('overlayKeySecondary'); - - this.ESC_BEHAVIOR = this.get('overviewEscBehavior'); - this.CLICK_EMPTY_CLOSE = this.get('clickEmptyClose'); - - this.FIX_NEW_WINDOW_FOCUS = this.get('newWindowFocusFix'); - this.FIX_NEW_WINDOW_MONITOR = this.get('newWindowMonitorFix'); - - this.HIGHLIGHTING_STYLE = this.get('highlightingStyle'); - this.HIGHLIGHT_DEFAULT = this.HIGHLIGHTING_STYLE === 0; - this.HIGHLIGHT_UNDERLINE = this.HIGHLIGHTING_STYLE === 1; - this.HIGHLIGHT_NONE = this.HIGHLIGHTING_STYLE === 2; - - this.DELAY_STARTUP = this.get('delayStartup'); - } - - _getAnimationDirection() { - if (this.ORIENTATION) - return this.WS_TMB_LEFT || !this.SHOW_WS_TMB ? 1 : 2; // 1 right, 2 left - else - return this.WS_TMB_TOP || !this.SHOW_WS_TMB ? 3 : 5; // 3 bottom, 5 top - } -}; diff --git a/extensions/46/vertical-workspaces/lib/swipeTracker.js b/extensions/46/vertical-workspaces/lib/swipeTracker.js deleted file mode 100644 index 560b296..0000000 --- a/extensions/46/vertical-workspaces/lib/swipeTracker.js +++ /dev/null @@ -1,116 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * swipeTracker.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import GObject from 'gi://GObject'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as SwipeTracker from 'resource:///org/gnome/shell/ui/swipeTracker.js'; - -let Me; -let opt; - -export const SwipeTrackerModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('swipeTrackerModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' SwipeTrackerModule - Keeping untouched'); - } - - _activateModule() { - if (opt.ORIENTATION) { // 1-VERTICAL, 0-HORIZONTAL - this._setVertical(); - } else { - this._setHorizontal(); - } - console.debug(' SwipeTrackerModule - Activated'); - } - - _disableModule() { - this._setHorizontal(); - - console.debug(' SwipeTrackerModule - Disabled'); - } - - _setVertical() { - // reverse swipe gestures for enter/leave overview and ws switching - Main.overview._swipeTracker.orientation = Clutter.Orientation.HORIZONTAL; - Main.wm._workspaceAnimation._swipeTracker.orientation = Clutter.Orientation.VERTICAL; - // overview's updateGesture() function should reflect ws tmb position to match appGrid/ws animation direction - // function in connection cannot be overridden in prototype of its class because connected is actually another copy of the original function - if (!this._originalGestureUpdateId) { - this._originalGestureUpdateId = GObject.signal_handler_find(Main.overview._swipeTracker._touchpadGesture, { signalId: 'update' }); - Main.overview._swipeTracker._touchpadGesture.block_signal_handler(this._originalGestureUpdateId); - Main.overview._swipeTracker._updateGesture = SwipeTrackerVertical._updateGesture; - this._vwGestureUpdateId = Main.overview._swipeTracker._touchpadGesture.connect('update', SwipeTrackerVertical._updateGesture.bind(Main.overview._swipeTracker)); - } - } - - _setHorizontal() { - // original swipeTrackers' orientation and updateGesture function - Main.overview._swipeTracker.orientation = Clutter.Orientation.VERTICAL; - Main.wm._workspaceAnimation._swipeTracker.orientation = Clutter.Orientation.HORIZONTAL; - Main.overview._swipeTracker._updateGesture = SwipeTracker.SwipeTracker.prototype._updateGesture; - if (this._vwGestureUpdateId) { - Main.overview._swipeTracker._touchpadGesture.disconnect(this._vwGestureUpdateId); - this._vwGestureUpdateId = 0; - } - if (this._originalGestureUpdateId) { - Main.overview._swipeTracker._touchpadGesture.unblock_signal_handler(this._originalGestureUpdateId); - this._originalGestureUpdateId = 0; - } - } -}; - -const SwipeTrackerVertical = { - _updateGesture(gesture, time, delta, distance) { - if (this._state !== 1) // State.SCROLLING) - return; - - if ((this._allowedModes & Main.actionMode) === 0 || !this.enabled) { - this._interrupt(); - return; - } - - if (opt.WS_TMB_RIGHT) - delta = -delta; - this._progress += delta / distance; - this._history.append(time, delta); - - this._progress = Math.clamp(this._progress, ...this._getBounds(this._initialProgress)); - this.emit('update', this._progress); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/util.js b/extensions/46/vertical-workspaces/lib/util.js deleted file mode 100644 index 0da67ce..0000000 --- a/extensions/46/vertical-workspaces/lib/util.js +++ /dev/null @@ -1,445 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * util.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import Gio from 'gi://Gio'; -import GLib from 'gi://GLib'; -import GObject from 'gi://GObject'; -import Meta from 'gi://Meta'; -import Shell from 'gi://Shell'; -import St from 'gi://St'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as ModalDialog from 'resource:///org/gnome/shell/ui/modalDialog.js'; -import { InjectionManager } from 'resource:///org/gnome/shell/extensions/extension.js'; - -let Me; -let _; -let _installedExtensions; - -export function init(me) { - Me = me; - _ = Me.gettext; -} - -export function cleanGlobals() { - Me = null; - _ = null; - _installedExtensions = null; -} - -export class Overrides extends InjectionManager { - constructor() { - super(); - this._overrides = {}; - } - - addOverride(name, prototype, overrideList) { - const backup = this.overrideProto(prototype, overrideList, name); - // don't update originals when override's just refreshing, keep initial content - let originals = this._overrides[name]?.originals; - if (!originals) - originals = backup; - this._overrides[name] = { - originals, - prototype, - }; - } - - removeOverride(name) { - const override = this._overrides[name]; - if (!override) - return false; - - this.overrideProto(override.prototype, override.originals, name); - delete this._overrides[name]; - return true; - } - - removeAll() { - for (let name in this._overrides) { - this.removeOverride(name); - delete this._overrides[name]; - } - } - - overrideProto(proto, overrides, name) { - const backup = {}; - const originals = this._overrides[name]?.originals; - for (let symbol in overrides) { - if (symbol.startsWith('after_')) { - const actualSymbol = symbol.slice('after_'.length); - let fn; - if (originals && originals[actualSymbol]) - fn = originals[actualSymbol]; - else - fn = proto[actualSymbol]; - const afterFn = overrides[symbol]; - proto[actualSymbol] = function (...args) { - args = Array.prototype.slice.call(args); - const res = fn.apply(this, args); - afterFn.apply(this, args); - return res; - }; - backup[actualSymbol] = fn; - } else if (overrides[symbol] !== null) { - backup[symbol] = proto[symbol]; - this._installMethod(proto, symbol, overrides[symbol]); - } - } - return backup; - } -} - -export function openPreferences(metadata) { - if (!metadata) - metadata = Me.metadata; - const windows = global.display.get_tab_list(Meta.TabList.NORMAL_ALL, null); - let tracker = Shell.WindowTracker.get_default(); - let metaWin, isMe = null; - - for (let win of windows) { - const app = tracker.get_window_app(win); - if (win.get_title()?.includes(metadata.name) && app.get_name() === 'Extensions') { - // this is our existing window - metaWin = win; - isMe = true; - break; - } else if (win.wm_class?.includes('org.gnome.Shell.Extensions')) { - // this is prefs window of another extension - metaWin = win; - isMe = false; - } - } - - if (metaWin && !isMe) { - // other prefs window blocks opening another prefs window, so close it - metaWin.delete(global.get_current_time()); - } else if (metaWin && isMe) { - // if prefs window already exist, move it to the current WS and activate it - metaWin.change_workspace(global.workspace_manager.get_active_workspace()); - metaWin.activate(global.get_current_time()); - } - - if (!metaWin || (metaWin && !isMe)) { - // delay to avoid errors if previous prefs window has been colsed - GLib.idle_add(GLib.PRIORITY_LOW, () => { - try { - Main.extensionManager.openExtensionPrefs(metadata.uuid, '', {}); - } catch (e) { - console.error(e); - } - }); - } -} - -export function activateSearchProvider(prefix = '') { - const searchEntry = Main.overview.searchEntry; - if (!searchEntry.get_text() || !searchEntry.get_text().startsWith(prefix)) { - prefix = `${prefix} `; - const position = prefix.length; - searchEntry.set_text(prefix); - searchEntry.get_first_child().set_cursor_position(position); - searchEntry.get_first_child().set_selection(position, position); - } else { - searchEntry.set_text(''); - } -} - -export function dashNotDefault() { - return Main.overview.dash !== Main.overview._overview._controls.layoutManager._dash; -} - -export function dashIsDashToDock() { - return Main.overview.dash._isHorizontal !== undefined; -} - -// Reorder Workspaces - callback for Dash and workspacesDisplay -export function reorderWorkspace(direction = 0) { - let activeWs = global.workspace_manager.get_active_workspace(); - let activeWsIdx = activeWs.index(); - let targetIdx = activeWsIdx + direction; - if (targetIdx > -1 && targetIdx < global.workspace_manager.get_n_workspaces()) - global.workspace_manager.reorder_workspace(activeWs, targetIdx); -} - -export function activateKeyboardForWorkspaceView() { - Main.ctrlAltTabManager._items.forEach(i => { - if (i.sortGroup === 1 && i.name === 'Windows') - Main.ctrlAltTabManager.focusGroup(i); - }); -} - -export function exposeWindows(adjustment, activateKeyboard) { - // expose windows for static overview modes - if (!adjustment.value && !Main.overview._animationInProgress) { - if (adjustment.value === 0) { - adjustment.value = 0; - adjustment.ease(1, { - duration: 200, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - if (activateKeyboard) { - Main.ctrlAltTabManager._items.forEach(i => { - if (i.sortGroup === 1 && i.name === 'Windows') - Main.ctrlAltTabManager.focusGroup(i); - }); - } - }, - }); - } - } -} - -export function isShiftPressed(state = null) { - if (state === null) - [,, state] = global.get_pointer(); - return (state & Clutter.ModifierType.SHIFT_MASK) !== 0; -} - -export function isCtrlPressed(state = null) { - if (state === null) - [,, state] = global.get_pointer(); - return (state & Clutter.ModifierType.CONTROL_MASK) !== 0; -} - -export function isAltPressed(state = null) { - if (state === null) - [,, state] = global.get_pointer(); - return (state & Clutter.ModifierType.MOD1_MASK) !== 0; -} - -export function fuzzyMatch(term, text) { - let pos = -1; - const matches = []; - // convert all accented chars to their basic form and to lower case - const _text = text.normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase(); - const _term = term.normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase(); - - // if term matches the substring exactly, gains the highest weight - if (_text.includes(_term)) - return 0; - - for (let i = 0; i < _term.length; i++) { - let c = _term[i]; - let p; - if (pos > 0) - p = _term[i - 1]; - while (true) { - pos += 1; - if (pos >= _text.length) - return -1; - - if (_text[pos] === c) { - matches.push(pos); - break; - } else if (_text[pos] === p) { - matches.pop(); - matches.push(pos); - } - } - } - - // add all position to get a weight of the result - // results closer to the beginning of the text and term characters closer to each other will gain more weight. - return matches.reduce((r, p) => r + p) - matches.length * matches[0] + matches[0]; -} - -export function strictMatch(term, text) { - // remove diacritics and accents from letters - let s = text.normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase(); - let p = term.normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase(); - let ps = p.split(/ +/); - - // allows to use multiple exact patterns separated by a space in arbitrary order - for (let w of ps) { // escape regex control chars - if (!s.match(w.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'))) - return -1; - } - return 0; -} - -export function isMoreRelevant(stringA, stringB, pattern) { - let regex = /[^a-zA-Z\d]/; - let strSplitA = stringA.normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase().split(regex); - let strSplitB = stringB.normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase().split(regex); - let aAny = false; - strSplitA.forEach(w => { - aAny = aAny || w.startsWith(pattern); - }); - let bAny = false; - strSplitB.forEach(w => { - bAny = bAny || w.startsWith(pattern); - }); - - // if both strings contain a word that starts with the pattern - // prefer the one whose first word starts with the pattern - if (aAny && bAny) - return !strSplitA[0].startsWith(pattern) && strSplitB[0].startsWith(pattern); - else - return !aAny && bAny; -} - -export function getEnabledExtensions(pattern = '') { - let result = []; - // extensionManager is unreliable at startup because it is uncertain whether all extensions have been loaded - // also gsettings key can contain already removed extensions (user deleted them without disabling them first) - // therefore we have to check what's really installed in the filesystem - if (!_installedExtensions) { - const extensionFiles = [...collectFromDatadirs('extensions', true)]; - _installedExtensions = extensionFiles.map(({ info }) => { - let fileType = info.get_file_type(); - if (fileType !== Gio.FileType.DIRECTORY) - return null; - const uuid = info.get_name(); - return uuid; - }); - } - // _enabledExtensions contains content of the enabled-extensions key from gsettings, not actual state - const enabled = Main.extensionManager._enabledExtensions; - result = _installedExtensions.filter(ext => enabled.includes(ext)); - // _extensions contains already loaded extensions, so we can try to filter out broken or incompatible extensions - const active = Main.extensionManager._extensions; - result = result.filter(ext => { - const extension = active.get(ext); - if (extension) - return ![3, 4].includes(extension.state); // 3 - ERROR, 4 - OUT_OF_TIME (not supported by shell-version in metadata) - // extension can be enabled but not yet loaded, we just cannot see its state at this moment, so let it pass as enabled - return true; - }); - // return only extensions matching the search pattern - return result.filter(uuid => uuid !== null && uuid.includes(pattern)); -} - -function* collectFromDatadirs(subdir, includeUserDir) { - let dataDirs = GLib.get_system_data_dirs(); - if (includeUserDir) - dataDirs.unshift(GLib.get_user_data_dir()); - - for (let i = 0; i < dataDirs.length; i++) { - let path = GLib.build_filenamev([dataDirs[i], 'gnome-shell', subdir]); - let dir = Gio.File.new_for_path(path); - - let fileEnum; - try { - fileEnum = dir.enumerate_children('standard::name,standard::type', - Gio.FileQueryInfoFlags.NONE, null); - } catch (e) { - fileEnum = null; - } - if (fileEnum !== null) { - let info; - while ((info = fileEnum.next_file(null))) - yield { dir: fileEnum.get_child(info), info }; - } - } -} - -export function getScrollDirection(event) { - // scroll wheel provides two types of direction information: - // 1. Clutter.ScrollDirection.DOWN / Clutter.ScrollDirection.UP - // 2. Clutter.ScrollDirection.SMOOTH + event.get_scroll_delta() - // first SMOOTH event returns 0 delta, - // so we need to always read event.direction - // since mouse without smooth scrolling provides exactly one SMOOTH event on one wheel rotation click - // on the other hand, under X11, one wheel rotation click sometimes doesn't send direction event, only several SMOOTH events - // so we also need to convert the delta to direction - let direction = event.get_scroll_direction(); - - if (direction !== Clutter.ScrollDirection.SMOOTH) - return direction; - - let [, delta] = event.get_scroll_delta(); - - if (!delta) - return null; - - direction = delta > 0 ? Clutter.ScrollDirection.DOWN : Clutter.ScrollDirection.UP; - - return direction; -} - -export function getWindows(workspace) { - // We ignore skip-taskbar windows in switchers, but if they are attached - // to their parent, their position in the MRU list may be more appropriate - // than the parent; so start with the complete list ... - let windows = global.display.get_tab_list(Meta.TabList.NORMAL_ALL, workspace); - // ... map windows to their parent where appropriate ... - return windows.map(w => { - return w.is_attached_dialog() ? w.get_transient_for() : w; - // ... and filter out skip-taskbar windows and duplicates - }).filter((w, i, a) => !w.skip_taskbar && a.indexOf(w) === i); -} - -export function monitorHasLowResolution(monitorIndex, resolutionLimit) { - resolutionLimit = resolutionLimit ?? 1200000; - monitorIndex = monitorIndex ?? global.display.get_primary_monitor(); - const monitorGeometry = global.display.get_monitor_geometry(monitorIndex); - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const monitorResolution = monitorGeometry.width * monitorGeometry.height; - return (monitorResolution / scaleFactor) < resolutionLimit; -} - -// ///////////////////////////////////////////////////////////////////////////////////////////// -// Status dialog that appears during updating V-Shell configuration and blocks inputs - -export const RestartMessage = GObject.registerClass({ - // Registered name should be unique - GTypeName: `RestartMessage${Math.floor(Math.random() * 1000)}`, -}, class RestartMessage extends ModalDialog.ModalDialog { - _init() { - super._init({ - shellReactive: false, - styleClass: 'restart-message headline update-message', - shouldFadeIn: false, - destroyOnClose: false, - }); - - const label = new St.Label({ - text: _('Updating V-Shell'), - x_align: Clutter.ActorAlign.CENTER, - y_align: Clutter.ActorAlign.CENTER, - }); - - this.contentLayout.add_child(label); - this.buttonLayout.hide(); - this.connect('destroy', () => this.removeMessage()); - } - - showMessage(timeout = 500) { - if (this._timeoutId || Me._resetInProgress || Main.layoutManager._startingUp) - return; - this._removeTimeout(); - this.open(); - this._timeoutId = GLib.timeout_add( - GLib.PRIORITY_LOW, - timeout, - () => { - this._timeoutId = 0; - this.removeMessage(); - return GLib.SOURCE_REMOVE; - } - ); - } - - _removeTimeout() { - if (this._timeoutId) { - GLib.source_remove(this._timeoutId); - this._timeoutId = 0; - } - } - - removeMessage() { - this._removeTimeout(); - this.close(); - } -}); diff --git a/extensions/46/vertical-workspaces/lib/windowAttentionHandler.js b/extensions/46/vertical-workspaces/lib/windowAttentionHandler.js deleted file mode 100644 index ae115ed..0000000 --- a/extensions/46/vertical-workspaces/lib/windowAttentionHandler.js +++ /dev/null @@ -1,185 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * windowAttentionHandler.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as MessageTray from 'resource:///org/gnome/shell/ui/messageTray.js'; - -const shellVersion46 = !Clutter.Container; - -let Me; -let opt; - -export const WindowAttentionHandlerModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('windowAttentionHandlerModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WindowAttentionHandlerModule - Keeping untouched'); - } - - _activateModule() { - this._updateConnections(); - console.debug(' WindowAttentionHandlerModule - Activated'); - } - - _disableModule() { - const reset = true; - this._updateConnections(reset); - - console.debug(' WindowAttentionHandlerModule - Disabled'); - } - - _updateConnections(reset) { - global.display.disconnectObject(Main.windowAttentionHandler); - - const handlerFnc = reset - ? Main.windowAttentionHandler._onWindowDemandsAttention - : WindowAttentionHandlerCommon._onWindowDemandsAttention; - - global.display.connectObject( - 'window-demands-attention', handlerFnc.bind(Main.windowAttentionHandler), - 'window-marked-urgent', handlerFnc.bind(Main.windowAttentionHandler), - Main.windowAttentionHandler); - } -}; - -const WindowAttentionHandlerCommon = { - _onWindowDemandsAttention(display, window) { - // Deny attention notifications if the App Grid is open, to avoid notification spree when opening a folder - if (Main.overview._shown && Main.overview.dash.showAppsButton.checked) { - return; - } else if (opt.WINDOW_ATTENTION_FOCUS_IMMEDIATELY) { - if (!Main.overview._shown) - Main.activateWindow(window); - return; - } - - const app = this._tracker.get_window_app(window); - let args; - if (shellVersion46) - args = { title: app.get_name() }; - else - args = app.get_name(); - - const source = new MessageTray.Source(args); - new Me.Util.Overrides().addOverride('MessageSource', source, WindowAttentionSourceCommon); - source._init(app, window); - Main.messageTray.add(source); - - let [title, body] = this._getTitleAndBanner(app, window); - args = shellVersion46 - ? [{ source, title, body, forFeedback: true }] - : [source, title, body]; - - const notification = new MessageTray.Notification(...args); - if (!shellVersion46) - notification.setForFeedback(true); - - notification.connect('activated', () => { - source.open(); - }); - - if (shellVersion46) { - notification.acknowledged = opt.WINDOW_ATTENTION_DISABLE_NOTIFICATIONS; - source.addNotification(notification); - if (opt.WINDOW_ATTENTION_DISABLE_NOTIFICATIONS) { - // just push the notification to the message tray without showing notification - notification.acknowledged = true; - Main.messageTray._notificationQueue.push(notification); - Main.panel.statusArea.dateMenu._indicator.show(); - } - window.connectObject('notify::title', () => { - [title, body] = this._getTitleAndBanner(app, window); - notification.set({ title, body }); - }, source); - } else { - if (opt.WINDOW_ATTENTION_DISABLE_NOTIFICATIONS) - // just push the notification to the message tray without showing notification - source.pushNotification(notification); - else - source.showNotification(notification); - - window.connectObject('notify::title', () => { - [title, body] = this._getTitleAndBanner(app, window); - notification.update(title, body); - }, source); - } - }, -}; - -const WindowAttentionSourceCommon = { - _init(app, window) { - this._window = window; - this._app = app; - - this._window.connectObject( - 'notify::demands-attention', this._sync.bind(this), - 'notify::urgent', this._sync.bind(this), - 'focus', () => this.destroy(), - 'unmanaged', () => this.destroy(), this); - }, - - _sync() { - if (this._window.demands_attention || this._window.urgent) - return; - this.destroy(); - }, - - _createPolicy() { - if (this._app && this._app.get_app_info()) { - let id = this._app.get_id().replace(/\.desktop$/, ''); - return new MessageTray.NotificationApplicationPolicy(id); - } else { - return new MessageTray.NotificationGenericPolicy(); - } - }, - - createIcon(size) { - return this._app.create_icon_texture(size); - }, - - destroy(params) { - this._window.disconnectObject(this); - - MessageTray.Source.prototype.destroy.bind(this)(params); - }, - - open() { - Main.activateWindow(this._window); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/windowManager.js b/extensions/46/vertical-workspaces/lib/windowManager.js deleted file mode 100644 index a6f9b09..0000000 --- a/extensions/46/vertical-workspaces/lib/windowManager.js +++ /dev/null @@ -1,380 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * windowManager.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import Meta from 'gi://Meta'; -import GObject from 'gi://GObject'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as WindowManager from 'resource:///org/gnome/shell/ui/windowManager.js'; -import * as WorkspaceAnimation from 'resource:///org/gnome/shell/ui/workspaceAnimation.js'; - -const MINIMIZE_WINDOW_ANIMATION_TIME = 400; // windowManager.MINIMIZE_WINDOW_ANIMATION_TIME -const MINIMIZE_WINDOW_ANIMATION_MODE = Clutter.AnimationMode.EASE_OUT_EXPO; // WindowManager.MINIMIZE_WINDOW_ANIMATION_MODE - -let Me; -let opt; - -export const WindowManagerModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - - this._originalMinimizeSigId = 0; - this._minimizeSigId = 0; - this._originalUnminimizeSigId = 0; - this._unminimizeSigId = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('windowManagerModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't even touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WindowManagerModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('WindowManager', WindowManager.WindowManager.prototype, WindowManagerCommon); - if (opt.WS_SWITCHER_CURRENT_MONITOR) - this._overrides.addOverride('WorkspaceAnimationController', WorkspaceAnimation.WorkspaceAnimationController.prototype, WorkspaceAnimationController); - - if (!this._minimizeSigId) { - this._originalMinimizeSigId = GObject.signal_handler_find(Main.wm._shellwm, { signalId: 'minimize' }); - if (this._originalMinimizeSigId) { - Main.wm._shellwm.block_signal_handler(this._originalMinimizeSigId); - this._minimizeSigId = Main.wm._shellwm.connect('minimize', WindowManagerCommon._minimizeWindow.bind(Main.wm)); - } - - this._originalUnminimizeSigId = GObject.signal_handler_find(Main.wm._shellwm, { signalId: 'unminimize' }); - if (this._originalUnminimizeSigId) { - Main.wm._shellwm.block_signal_handler(this._originalUnminimizeSigId); - this._unminimizeSigId = Main.wm._shellwm.connect('unminimize', WindowManagerCommon._unminimizeWindow.bind(Main.wm)); - } - } - console.debug(' WindowManagerModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - if (this._minimizeSigId) { - Main.wm._shellwm.disconnect(this._minimizeSigId); - this._minimizeSigId = 0; - } - if (this._originalMinimizeSigId) { - Main.wm._shellwm.unblock_signal_handler(this._originalMinimizeSigId); - this._originalMinimizeSigId = 0; - } - - if (this._unminimizeSigId) { - Main.wm._shellwm.disconnect(this._unminimizeSigId); - this._unminimizeSigId = 0; - } - if (this._originalUnminimizeSigId) { - Main.wm._shellwm.unblock_signal_handler(this._originalUnminimizeSigId); - this._originalUnminimizeSigId = 0; - } - - console.debug(' WindowManagerModule - Disabled'); - } -}; - -const WindowManagerCommon = { - actionMoveWorkspace(workspace) { - if (!Main.sessionMode.hasWorkspaces) - return; - - if (opt.WS_SWITCHER_CURRENT_MONITOR) - this._switchWorkspaceCurrentMonitor(workspace); - else if (!workspace.active) - workspace.activate(global.get_current_time()); - }, - - actionMoveWindow(window, workspace) { - if (!Main.sessionMode.hasWorkspaces) - return; - - if (!workspace.active) { - // This won't have any effect for "always sticky" windows - // (like desktop windows or docks) - - this._workspaceAnimation.movingWindow = window; - window.change_workspace(workspace); - - global.display.clear_mouse_mode(); - - if (opt.SWITCH_ONLY_CURRENT_MONITOR_WS) { - this._switchWorkspaceCurrentMonitor(workspace, window.get_monitor()); - window.activate(global.get_current_time()); - } else { - workspace.activate_with_focus(window, global.get_current_time()); - } - } - }, - - _switchWorkspaceCurrentMonitor(workspace, monitor) { - // const focusedWindow = global.display.get_focus_window(); - // const currentMonitor = focusedWindow ? focusedWindow.get_monitor() : global.display.get_current_monitor(); - // using focused window to determine the current monitor can lead to inconsistent behavior and switching monitors between switches - // depending on which window takes focus on each workspace - // mouse pointer is more stable and predictable source - const currentMonitor = monitor ? monitor : global.display.get_current_monitor(); - const primaryMonitor = currentMonitor === Main.layoutManager.primaryIndex; - const nMonitors = Main.layoutManager.monitors.length; - const lastIndexCorrection = Meta.prefs_get_dynamic_workspaces() ? 2 : 1; - const lastIndex = global.workspaceManager.get_n_workspaces() - lastIndexCorrection; - const targetWsIndex = workspace.index(); - const activeWs = global.workspaceManager.get_active_workspace(); - const activeWsIndex = activeWs.index(); - const diff = activeWsIndex - targetWsIndex; - - let direction = diff > 0 ? Meta.MotionDirection.UP : Meta.MotionDirection.DOWN; - if (diff === 0) { - // no actual ws to switch, but secondary monitors are always in wraparound mode so we need to get direction - direction = activeWsIndex >= lastIndex ? Meta.MotionDirection.DOWN : Meta.MotionDirection.UP; - } - if (Math.abs(diff) > 1) { - // workspace is probably in wraparound mode and just wrapped so so we need to translate direction - direction = diff > 0 ? Meta.MotionDirection.DOWN : Meta.MotionDirection.UP; - } - - if (!primaryMonitor) { - this._rotateWorkspaces(direction, currentMonitor); - return; - } - - // avoid ws rotations if the last empty dynamic workspace is involved, but allow to rotate from the last to the first, if wraparound is enabled - if (workspace !== activeWs && !((targetWsIndex > lastIndex && direction === Meta.MotionDirection.DOWN) || (activeWsIndex > lastIndex && targetWsIndex >= lastIndex))) { - for (let i = 0; i < nMonitors; i++) { - if (i !== currentMonitor) { - const oppositeDirection = direction === Meta.MotionDirection.UP ? Meta.MotionDirection.DOWN : Meta.MotionDirection.UP; - this._rotateWorkspaces(oppositeDirection, i); - } - } - } - workspace.activate(global.get_current_time()); - }, - - _rotateWorkspaces(direction = 0, monitorIndex = -1, step = 1) { - step = direction === Meta.MotionDirection.UP ? Number(step) : -step; - const monitor = monitorIndex > -1 ? monitorIndex : global.display.get_current_monitor(); - // don't move windows to the last empty workspace if dynamic workspaces are enabled - const lastIndexCorrection = Meta.prefs_get_dynamic_workspaces() ? 2 : 1; - const lastIndex = global.workspaceManager.get_n_workspaces() - lastIndexCorrection; - let windows = Me.Util.getWindows(null); - for (let win of windows.reverse()) { - // avoid moving modal windows as they move with their parents (and vice versa) immediately, before we move the parent window. - if (win.get_monitor() === monitor && !win.is_always_on_all_workspaces() && !win.is_attached_dialog() && !win.get_transient_for()) { - let wWs = win.get_workspace().index(); - wWs += step; - if (wWs < 0) - wWs = lastIndex; - if (wWs > lastIndex) - wWs = 0; - const ws = global.workspaceManager.get_workspace_by_index(wWs); - win.change_workspace(ws); - } - } - }, - - // fix for mainstream bug - fullscreen windows should minimize using opacity transition - // but its being applied directly on window actor and that doesn't work - // anyway, animation is better, even if the Activities button is not visible... - // and also add support for bottom position of the panel - _minimizeWindow(shellwm, actor) { - const types = [ - Meta.WindowType.NORMAL, - Meta.WindowType.MODAL_DIALOG, - Meta.WindowType.DIALOG, - ]; - if (!this._shouldAnimateActor(actor, types)) { - shellwm.completed_minimize(actor); - return; - } - - actor.set_scale(1.0, 1.0); - - this._minimizing.add(actor); - - /* if (actor.meta_window.is_monitor_sized()) { - actor.get_first_child().ease({ - opacity: 0, - duration: MINIMIZE_WINDOW_ANIMATION_TIME, - mode: MINIMIZE_WINDOW_ANIMATION_MODE, - onStopped: () => this._minimizeWindowDone(shellwm, actor), - }); - } else { */ - let xDest, yDest, xScale, yScale; - let [success, geom] = actor.meta_window.get_icon_geometry(); - if (success) { - xDest = geom.x; - yDest = geom.y; - xScale = geom.width / actor.width; - yScale = geom.height / actor.height; - } else { - let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()]; - if (!monitor) { - this._minimizeWindowDone(); - return; - } - xDest = monitor.x; - yDest = opt.PANEL_POSITION_TOP ? monitor.y : monitor.y + monitor.height; - if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) - xDest += monitor.width; - xScale = 0; - yScale = 0; - } - - actor.ease({ - scale_x: xScale, - scale_y: yScale, - x: xDest, - y: yDest, - duration: MINIMIZE_WINDOW_ANIMATION_TIME, - mode: MINIMIZE_WINDOW_ANIMATION_MODE, - onStopped: () => this._minimizeWindowDone(shellwm, actor), - }); - // } - }, - - _minimizeWindowDone(shellwm, actor) { - if (this._minimizing.delete(actor)) { - actor.remove_all_transitions(); - actor.set_scale(1.0, 1.0); - actor.get_first_child().set_opacity(255); - actor.set_pivot_point(0, 0); - - shellwm.completed_minimize(actor); - } - }, - - _unminimizeWindow(shellwm, actor) { - const types = [ - Meta.WindowType.NORMAL, - Meta.WindowType.MODAL_DIALOG, - Meta.WindowType.DIALOG, - ]; - if (!this._shouldAnimateActor(actor, types)) { - shellwm.completed_unminimize(actor); - return; - } - - this._unminimizing.add(actor); - - /* if (false/* actor.meta_window.is_monitor_sized()) { - actor.opacity = 0; - actor.set_scale(1.0, 1.0); - actor.ease({ - opacity: 255, - duration: MINIMIZE_WINDOW_ANIMATION_TIME, - mode: MINIMIZE_WINDOW_ANIMATION_MODE, - onStopped: () => this._unminimizeWindowDone(shellwm, actor), - }); - } else { */ - let [success, geom] = actor.meta_window.get_icon_geometry(); - if (success) { - actor.set_position(geom.x, geom.y); - actor.set_scale(geom.width / actor.width, - geom.height / actor.height); - } else { - let monitor = Main.layoutManager.monitors[actor.meta_window.get_monitor()]; - if (!monitor) { - actor.show(); - this._unminimizeWindowDone(); - return; - } - actor.set_position(monitor.x, opt.PANEL_POSITION_TOP ? monitor.y : monitor.y + monitor.height); - if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) - actor.x += monitor.width; - actor.set_scale(0, 0); - } - - let rect = actor.meta_window.get_buffer_rect(); - let [xDest, yDest] = [rect.x, rect.y]; - - actor.show(); - actor.ease({ - scale_x: 1, - scale_y: 1, - x: xDest, - y: yDest, - duration: MINIMIZE_WINDOW_ANIMATION_TIME, - mode: MINIMIZE_WINDOW_ANIMATION_MODE, - onStopped: () => this._unminimizeWindowDone(shellwm, actor), - }); - // } - }, -}; - -const WorkspaceAnimationController = { - _prepareWorkspaceSwitch(workspaceIndices) { - if (this._switchData) - return; - - const workspaceManager = global.workspace_manager; - const nWorkspaces = workspaceManager.get_n_workspaces(); - - const switchData = {}; - - this._switchData = switchData; - switchData.monitors = []; - - switchData.gestureActivated = false; - switchData.inProgress = false; - - if (!workspaceIndices) - workspaceIndices = [...Array(nWorkspaces).keys()]; - - let monitors = opt.WS_SWITCHER_CURRENT_MONITOR - ? [Main.layoutManager.currentMonitor] : Main.layoutManager.monitors; - monitors = Meta.prefs_get_workspaces_only_on_primary() - ? [Main.layoutManager.primaryMonitor] : monitors; - - for (const monitor of monitors) { - if (Meta.prefs_get_workspaces_only_on_primary() && - monitor.index !== Main.layoutManager.primaryIndex) - continue; - - const group = new WorkspaceAnimation.MonitorGroup(monitor, workspaceIndices, this.movingWindow); - - Main.uiGroup.insert_child_above(group, global.window_group); - - switchData.monitors.push(group); - } - - Meta.disable_unredirect_for_display(global.display); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/windowPreview.js b/extensions/46/vertical-workspaces/lib/windowPreview.js deleted file mode 100644 index a529dc1..0000000 --- a/extensions/46/vertical-workspaces/lib/windowPreview.js +++ /dev/null @@ -1,629 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * windowPreview.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import GLib from 'gi://GLib'; -import Clutter from 'gi://Clutter'; -import St from 'gi://St'; -import Meta from 'gi://Meta'; -import Shell from 'gi://Shell'; -import Pango from 'gi://Pango'; -import Graphene from 'gi://Graphene'; -import Atk from 'gi://Atk'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as DND from 'resource:///org/gnome/shell/ui/dnd.js'; -import * as OverviewControls from 'resource:///org/gnome/shell/ui/overviewControls.js'; -import * as WindowPreview from 'resource:///org/gnome/shell/ui/windowPreview.js'; - -let Me; -let opt; - -const WINDOW_SCALE_TIME = 200; -const WINDOW_ACTIVE_SIZE_INC = 5; -const WINDOW_OVERLAY_FADE_TIME = 200; -const WINDOW_DND_SIZE = 256; -const DRAGGING_WINDOW_OPACITY = 100; -const ICON_OVERLAP = 0.7; -const ICON_TITLE_SPACING = 6; - -const ControlsState = OverviewControls.ControlsState; - -export const WindowPreviewModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('windowPreviewModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WindowPreviewModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('WindowPreview', WindowPreview.WindowPreview.prototype, WindowPreviewCommon); - // A shorter timeout allows user to quickly cancel the selection by leaving the preview with the mouse pointer - // if (opt.ALWAYS_ACTIVATE_SELECTED_WINDOW) - // WindowPreview.WINDOW_OVERLAY_IDLE_HIDE_TIMEOUT = 150; // incompatible - console.debug(' WindowPreviewModule - Activated'); - } - - _disableModule() { - // If WindowPreview._init was injected by another extension (like Burn My Windows) - // which enables/disables before V-Shell - // don't restore the original if it's not injected, - // because it would restore injected _init and recursion would freeze GS when extensions are enabled again. - // This can happen when all extension re-enabled, not only when screen is locked/unlocked - // If _init doesn't include "fn.apply(this, args)" when reset === true, some extension already restored the original - const skipReset = WindowPreview.WindowPreview.prototype._init.toString().includes('fn.apply(this, args)'); - if (this._overrides && skipReset) { - // skip restoring original _init() - this._overrides['_init'] = null; - } - - if (this._overrides) - this._overrides.removeAll(); - - this._overrides = null; - - console.debug(' WindowPreviewModule - Disabled'); - } -}; - -const WindowPreviewCommon = { - _init(metaWindow, workspace, overviewAdjustment) { - this.metaWindow = metaWindow; - this.metaWindow._delegate = this; - this._windowActor = metaWindow.get_compositor_private(); - this._workspace = workspace; - this._overviewAdjustment = overviewAdjustment; - - const ICON_SIZE = opt.WIN_PREVIEW_ICON_SIZE; - - Shell.WindowPreview.prototype._init.bind(this)({ - reactive: true, - can_focus: true, - accessible_role: Atk.Role.PUSH_BUTTON, - offscreen_redirect: Clutter.OffscreenRedirect.AUTOMATIC_FOR_OPACITY, - }); - - const windowContainer = new Clutter.Actor({ - pivot_point: new Graphene.Point({ x: 0.5, y: 0.5 }), - }); - this.window_container = windowContainer; - - windowContainer.connect('notify::scale-x', - () => this._adjustOverlayOffsets()); - // gjs currently can't handle setting an actors layout manager during - // the initialization of the actor if that layout manager keeps track - // of its container, so set the layout manager after creating the - // container - windowContainer.layout_manager = new Shell.WindowPreviewLayout(); - this.add_child(windowContainer); - - this._addWindow(metaWindow); - - this._delegate = this; - - this._stackAbove = null; - - this._cachedBoundingBox = { - x: windowContainer.layout_manager.bounding_box.x1, - y: windowContainer.layout_manager.bounding_box.y1, - width: windowContainer.layout_manager.bounding_box.get_width(), - height: windowContainer.layout_manager.bounding_box.get_height(), - }; - - windowContainer.layout_manager.connect( - 'notify::bounding-box', layout => { - this._cachedBoundingBox = { - x: layout.bounding_box.x1, - y: layout.bounding_box.y1, - width: layout.bounding_box.get_width(), - height: layout.bounding_box.get_height(), - }; - - // A bounding box of 0x0 means all windows were removed - if (layout.bounding_box.get_area() > 0) - this.emit('size-changed'); - }); - - this._windowActor.connectObject('destroy', () => this.destroy(), this); - - this._updateAttachedDialogs(); - - let clickAction = new Clutter.ClickAction(); - clickAction.connect('clicked', act => { - const button = act.get_button(); - if (button === Clutter.BUTTON_SECONDARY) { - if (opt.WIN_PREVIEW_SEC_BTN_ACTION === 1) { - this._closeWinAction(); - return Clutter.EVENT_STOP; - } else if (opt.WIN_PREVIEW_SEC_BTN_ACTION === 2) { - this._searchAppWindowsAction(); - return Clutter.EVENT_STOP; - } else if (opt.WIN_PREVIEW_SEC_BTN_ACTION === 3 && global.windowThumbnails) { - this._removeLaters(); - global.windowThumbnails?.createThumbnail(metaWindow); - return Clutter.EVENT_STOP; - } - } else if (button === Clutter.BUTTON_MIDDLE) { - if (opt.WIN_PREVIEW_MID_BTN_ACTION === 1) { - this._closeWinAction(); - return Clutter.EVENT_STOP; - } else if (opt.WIN_PREVIEW_MID_BTN_ACTION === 2) { - this._searchAppWindowsAction(); - return Clutter.EVENT_STOP; - } else if (opt.WIN_PREVIEW_SEC_BTN_ACTION === 3 && global.windowThumbnails) { - this._removeLaters(); - global.windowThumbnails?.createThumbnail(metaWindow); - return Clutter.EVENT_STOP; - } - } - return this._activate(); - }); - - - if (this._onLongPress) { - clickAction.connect('long-press', this._onLongPress.bind(this)); - } else { - clickAction.connect('long-press', (action, actor, state) => { - if (state === Clutter.LongPressState.ACTIVATE) - this.showOverlay(true); - return true; - }); - } - - this.connect('destroy', this._onDestroy.bind(this)); - - this._draggable = DND.makeDraggable(this, { - restoreOnSuccess: true, - manualMode: !!this._onLongPress, - dragActorMaxSize: WINDOW_DND_SIZE, - dragActorOpacity: DRAGGING_WINDOW_OPACITY, - }); - - // _draggable.addClickAction is new in GS45 - if (this._draggable.addClickAction) - this._draggable.addClickAction(clickAction); - else - this.add_action(clickAction); - - this._draggable.connect('drag-begin', this._onDragBegin.bind(this)); - this._draggable.connect('drag-cancelled', this._onDragCancelled.bind(this)); - this._draggable.connect('drag-end', this._onDragEnd.bind(this)); - this.inDrag = false; - - this._selected = false; - this._overlayEnabled = true; - this._overlayShown = false; - this._closeRequested = false; - this._idleHideOverlayId = 0; - - const tracker = Shell.WindowTracker.get_default(); - const app = tracker.get_window_app(this.metaWindow); - this._icon = app.create_icon_texture(ICON_SIZE); - this._icon.add_style_class_name('icon-dropshadow'); - this._icon.set({ - reactive: true, - pivot_point: new Graphene.Point({ x: 0.5, y: 0.5 }), - }); - this._icon.add_constraint(new Clutter.BindConstraint({ - source: windowContainer, - coordinate: Clutter.BindCoordinate.POSITION, - })); - this._icon.add_constraint(new Clutter.AlignConstraint({ - source: windowContainer, - align_axis: Clutter.AlignAxis.X_AXIS, - factor: 0.5, - })); - this._icon.add_constraint(new Clutter.AlignConstraint({ - source: windowContainer, - align_axis: Clutter.AlignAxis.Y_AXIS, - pivot_point: new Graphene.Point({ x: -1, y: ICON_OVERLAP }), - factor: 1, - })); - - if (opt.WINDOW_ICON_CLICK_ACTION) { - const iconClickAction = new Clutter.ClickAction(); - iconClickAction.connect('clicked', act => { - if (act.get_button() === Clutter.BUTTON_PRIMARY) { - if (opt.WINDOW_ICON_CLICK_ACTION === 1) { - this._searchAppWindowsAction(); - return Clutter.EVENT_STOP; - } else if (opt.WINDOW_ICON_CLICK_ACTION === 2 && global.windowThumbnails) { - this._removeLaters(); - global.windowThumbnails?.createThumbnail(metaWindow); - return Clutter.EVENT_STOP; - } - } /* else if (act.get_button() === Clutter.BUTTON_SECONDARY) { - return Clutter.EVENT_STOP; - }*/ - return Clutter.EVENT_PROPAGATE; - }); - this._icon.add_action(iconClickAction); - } - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - this._title = new St.Label({ - visible: false, - style_class: 'window-caption', - text: this._getCaption(), - reactive: true, - }); - this._title.clutter_text.single_line_mode = true; - this._title.add_constraint(new Clutter.BindConstraint({ - source: windowContainer, - coordinate: Clutter.BindCoordinate.X, - })); - - let offset; - if (opt.WIN_TITLES_POSITION < 2) { - // we cannot get proper title height before it gets to the stage, so 35 is estimated height + spacing - offset = -scaleFactor * (ICON_SIZE * ICON_OVERLAP + 35); - } else { - offset = scaleFactor * (ICON_SIZE * (1 - ICON_OVERLAP) + 4); - } - this._title.add_constraint(new Clutter.BindConstraint({ - source: windowContainer, - coordinate: Clutter.BindCoordinate.Y, - offset, - })); - this._title.add_constraint(new Clutter.AlignConstraint({ - source: windowContainer, - align_axis: Clutter.AlignAxis.X_AXIS, - factor: 0.5, - })); - this._title.add_constraint(new Clutter.AlignConstraint({ - source: windowContainer, - align_axis: Clutter.AlignAxis.Y_AXIS, - pivot_point: new Graphene.Point({ x: -1, y: 0 }), - factor: 1, - })); - this._title.clutter_text.ellipsize = Pango.EllipsizeMode.END; - this.label_actor = this._title; - this.metaWindow.connectObject( - 'notify::title', () => (this._title.text = this._getCaption()), - this); - - const layout = Meta.prefs_get_button_layout(); - this._closeButtonSide = - layout.left_buttons.includes(Meta.ButtonFunction.CLOSE) - ? St.Side.LEFT : St.Side.RIGHT; - this._closeButton = new St.Button({ - visible: false, - style_class: 'window-close', - icon_name: 'preview-close-symbolic', - }); - this._closeButton.add_constraint(new Clutter.BindConstraint({ - source: windowContainer, - coordinate: Clutter.BindCoordinate.POSITION, - })); - this._closeButton.add_constraint(new Clutter.AlignConstraint({ - source: windowContainer, - align_axis: Clutter.AlignAxis.X_AXIS, - pivot_point: new Graphene.Point({ x: 0.5, y: -1 }), - factor: this._closeButtonSide === St.Side.LEFT ? 0 : 1, - })); - this._closeButton.add_constraint(new Clutter.AlignConstraint({ - source: windowContainer, - align_axis: Clutter.AlignAxis.Y_AXIS, - pivot_point: new Graphene.Point({ x: -1, y: 0.5 }), - factor: 0, - })); - this._closeButton.connect('clicked', () => this._deleteAll()); - - this.add_child(this._title); - this.add_child(this._icon); - this.add_child(this._closeButton); - - this._overviewAdjustment.connectObject( - 'notify::value', () => this._updateIconScale(), this); - this._updateIconScale(); - - this.connect('notify::realized', () => { - if (!this.realized) - return; - - this._title.ensure_style(); - this._icon.ensure_style(); - }); - - if (ICON_SIZE < 22) { - // disable app icon - this._icon.hide(); - } else { - this._updateIconScale(); - } - - - - // if window is created while the overview is shown, icon and title should be visible immediately - if (Main.overview._overview._controls._stateAdjustment.value < 1) { - this._icon.scale_x = 0; - this._icon.scale_y = 0; - this._title.opacity = 0; - } - - if (opt.ALWAYS_SHOW_WIN_TITLES) - this._title.show(); - - if (opt.OVERVIEW_MODE === 1) { - // spread windows on hover - this._wsStateConId = this.connect('enter-event', () => { - // don't spread windows if user don't use pointer device at this moment - if (global.get_pointer()[0] === opt.showingPointerX || Main.overview._overview._controls._stateAdjustment.value < 1) - return; - - opt.WORKSPACE_MODE = 1; - const view = this._workspace.get_parent(); - view.exposeWindows(this._workspace.metaWorkspace.index()); - this.disconnect(this._wsStateConId); - }); - } - - if (opt.OVERVIEW_MODE) { - // show window icon and title on ws windows spread - this._stateAdjustmentSigId = this._workspace.stateAdjustment.connect('notify::value', this._updateIconScale.bind(this)); - } - - const metaWin = this.metaWindow; - if (opt.DASH_ISOLATE_WS && !metaWin._wsChangedConId) { - metaWin._wsChangedConId = metaWin.connect('workspace-changed', - () => Main.overview.dash._queueRedisplay()); - } else if (!opt.DASH_ISOLATE_WS && metaWin._wsChangedConId) { - metaWin.disconnect(metaWin._wsChangedConId); - } - }, - - _closeWinAction() { - this.hide(); - this._deleteAll(); - }, - - _removeLaters() { - if (this._longPressLater) { - const laters = global.compositor.get_laters(); - laters.remove(this._longPressLater); - delete this._longPressLater; - } - }, - - _searchAppWindowsAction() { - // this action cancels long-press event and the 'long-press-cancel' event is used by the Shell to actually initiate DnD - // so the dnd initiation needs to be removed - this._removeLaters(); - const tracker = Shell.WindowTracker.get_default(); - const appName = tracker.get_window_app(this.metaWindow).get_name(); - Me.Util.activateSearchProvider(`${Me.WSP_PREFIX} ${appName}`); - }, - - _updateIconScale() { - let { currentState, initialState, finalState } = - this._overviewAdjustment.getStateTransitionParams(); - - // Current state - 0 - HIDDEN, 1 - WINDOW_PICKER, 2 - APP_GRID - const primaryMonitor = this.metaWindow.get_monitor() === global.display.get_primary_monitor(); - - const visible = - (initialState > ControlsState.HIDDEN || finalState > ControlsState.HIDDEN) && - !(finalState === ControlsState.APP_GRID && opt.WS_ANIMATION && primaryMonitor); - - let scale = 0; - if (visible) - scale = currentState >= 1 ? 1 : currentState % 1; - - if (!primaryMonitor && opt.WORKSPACE_MODE && - ((initialState === ControlsState.WINDOW_PICKER && finalState === ControlsState.APP_GRID) || - (initialState === ControlsState.APP_GRID && finalState === ControlsState.WINDOW_PICKER)) - ) - scale = 1; - else if (!primaryMonitor && opt.OVERVIEW_MODE && !opt.WORKSPACE_MODE) - scale = 0; - /* } else if (primaryMonitor && ((initialState === ControlsState.WINDOW_PICKER && finalState === ControlsState.APP_GRID) || - initialState === ControlsState.APP_GRID && finalState === ControlsState.HIDDEN)) {*/ - else if (primaryMonitor && currentState > ControlsState.WINDOW_PICKER) - scale = 0; - - // in static workspace mode show icon and title on windows expose - if (opt.OVERVIEW_MODE) { - if (currentState === 1) - scale = this._workspace._background._stateAdjustment.value; - else if ((finalState === 1 && !opt.WORKSPACE_MODE) || (finalState === 0 && !opt.WORKSPACE_MODE)) - return; - } - - if (!opt.WS_ANIMATION && (Main.overview.searchController.searchActive || - ((initialState === ControlsState.WINDOW_PICKER && finalState === ControlsState.APP_GRID) || - (initialState === ControlsState.APP_GRID && finalState === ControlsState.WINDOW_PICKER))) - ) - return; - - // if titles are in 'always show' mode, we need to add transition between visible/invisible state - // but the transition is quite expensive, - // showing the titles at the end of the transition is good enough and workspace preview transition is much smoother - if (scale === 1) { - this._icon.set({ - scale_x: 1, - scale_y: 1, - }); - this._title.ease({ - duration: 100, - opacity: 255, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } else { - this._title.opacity = 0; - this._icon.set({ - scale_x: scale, - scale_y: scale, - }); - } - }, - - showOverlay(animate) { - if (!this._overlayEnabled) - return; - - if (this._overlayShown) - return; - - this._overlayShown = true; - if (opt.WIN_TITLES_POSITION === 2) - this._restack(); - - // If we're supposed to animate and an animation in our direction - // is already happening, let that one continue - const ongoingTransition = this._title.get_transition('opacity'); - if (animate && - ongoingTransition && - ongoingTransition.get_interval().peek_final_value() === 255) - return; - - const toShow = this._windowCanClose() && opt.SHOW_CLOSE_BUTTON - ? [this._closeButton] - : []; - - if (!opt.ALWAYS_SHOW_WIN_TITLES) - toShow.push(this._title); - - - toShow.forEach(a => { - a.opacity = 0; - a.show(); - a.ease({ - opacity: 255, - duration: animate ? WINDOW_OVERLAY_FADE_TIME : 0, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - }); - - const [width, height] = this.window_container.get_size(); - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const activeExtraSize = WINDOW_ACTIVE_SIZE_INC * 2 * scaleFactor; - const origSize = Math.max(width, height); - const scale = (origSize + activeExtraSize) / origSize; - - this.window_container.ease({ - scale_x: scale, - scale_y: scale, - duration: animate ? WINDOW_SCALE_TIME : 0, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - - this.emit('show-chrome'); - }, - - hideOverlay(animate) { - if (!this._overlayShown) - return; - this._overlayShown = false; - - if (opt.ALWAYS_ACTIVATE_SELECTED_WINDOW && Main.overview._overview.controls._stateAdjustment.value < 1) - this._activateSelected = true; - - - if (opt.WIN_TITLES_POSITION === 2) - this._restack(); - - // If we're supposed to animate and an animation in our direction - // is already happening, let that one continue - const ongoingTransition = this._title.get_transition('opacity'); - if (animate && - ongoingTransition && - ongoingTransition.get_interval().peek_final_value() === 0) - return; - - const toHide = [this._closeButton]; - - if (!opt.ALWAYS_SHOW_WIN_TITLES) - toHide.push(this._title); - - toHide.forEach(a => { - a.opacity = 255; - a.ease({ - opacity: 0, - duration: animate ? WINDOW_OVERLAY_FADE_TIME : 0, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => a.hide(), - }); - }); - - if (this.window_container) { - this.window_container.ease({ - scale_x: 1, - scale_y: 1, - duration: animate ? WINDOW_SCALE_TIME : 0, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - }, - - overlapHeights() { - const [, titleHeight] = this._title.get_preferred_height(-1); - - const topOverlap = 0; - const bottomOverlap = opt.WIN_TITLES_POSITION === 2 ? titleHeight + ICON_TITLE_SPACING : 0; - - return [topOverlap, bottomOverlap]; - }, - - _onDestroy() { - if (this._activateSelected) - this._activate(); - - this.metaWindow._delegate = null; - this._delegate = null; - this._destroyed = true; - - if (this._longPressLater) { - const laters = global.compositor.get_laters(); - laters.remove(this._longPressLater); - delete this._longPressLater; - } - - if (this._idleHideOverlayId > 0) { - GLib.source_remove(this._idleHideOverlayId); - this._idleHideOverlayId = 0; - } - - if (this.inDrag) { - this.emit('drag-end'); - this.inDrag = false; - } - - if (this._stateAdjustmentSigId) - this._workspace.stateAdjustment.disconnect(this._stateAdjustmentSigId); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/workspace.js b/extensions/46/vertical-workspaces/lib/workspace.js deleted file mode 100644 index 9f1dbbc..0000000 --- a/extensions/46/vertical-workspaces/lib/workspace.js +++ /dev/null @@ -1,472 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * workspace.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import St from 'gi://St'; -// import Graphene from 'gi://Graphene'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Workspace from 'resource:///org/gnome/shell/ui/workspace.js'; -import * as Params from 'resource:///org/gnome/shell/misc/params.js'; -import * as Util from 'resource:///org/gnome/shell/misc/util.js'; - -let Me; -let opt; - -let WINDOW_PREVIEW_MAXIMUM_SCALE = 0.95; - -export const WorkspaceModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('workspaceModule'); - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WorkspaceModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('WorkspaceBackground', Workspace.WorkspaceBackground.prototype, WorkspaceBackground); - - // fix overlay base for Vertical Workspaces - this._overrides.addOverride('WorkspaceLayout', Workspace.WorkspaceLayout.prototype, WorkspaceLayout); - console.debug(' WorkspaceModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - console.debug(' WorkspaceModule - Disabled'); - } - - setWindowPreviewMaxScale(scale) { - WINDOW_PREVIEW_MAXIMUM_SCALE = scale; - } -}; - -// workaround for upstream bug (that is not that invisible in default shell) -// smaller window cannot be scaled below 0.95 (WINDOW_PREVIEW_MAXIMUM_SCALE) -// when its target scale for exposed windows view (workspace state 1) is bigger than the scale needed for ws state 0. -// in workspace state 0 where windows are not spread and window scale should follow workspace scale, -// this window follows proper top left corner position, but doesn't scale with the workspace -// so it looks bad and the window can exceed border of the workspace -// extremely annoying in OVERVIEW_MODE 1 with single smaller window on the workspace, also affects appGrid transition animation - -// disadvantage of following workaround - the WINDOW_PREVIEW_MAXIMUM_SCALE value is common for every workspace, -// on multi-monitor system can be visible unwanted scaling of windows on workspace in WORKSPACE_MODE 0 (windows not spread) -// when leaving overview while any other workspace is in the WORKSPACE_MODE 1. -const WorkspaceLayout = { - // injection to _init() - after__init() { - if (opt.OVERVIEW_MODE !== 1) - WINDOW_PREVIEW_MAXIMUM_SCALE = 0.95; - if (opt.OVERVIEW_MODE === 1) { - this._stateAdjustment.connect('notify::value', () => { - // scale 0.1 for window state 0 just needs to be smaller then possible scale of any window in spread view - const scale = this._stateAdjustment.value ? 0.95 : 0.1; - if (scale !== WINDOW_PREVIEW_MAXIMUM_SCALE) { - WINDOW_PREVIEW_MAXIMUM_SCALE = scale; - // when transition to ws state 1 (WINDOW_PICKER) begins, replace the constant with the original one - // and force recalculation of the target layout, so the transition will be smooth - this._needsLayout = true; - } - }); - } - }, - - _adjustSpacingAndPadding(rowSpacing, colSpacing, containerBox) { - if (this._sortedWindows.length === 0) - return [rowSpacing, colSpacing, containerBox]; - - // All of the overlays have the same chrome sizes, - // so just pick the first one. - const window = this._sortedWindows[0]; - - const [topOversize, bottomOversize] = window.chromeHeights(); - const [leftOversize, rightOversize] = window.chromeWidths(); - - let oversize = Math.max(topOversize, bottomOversize, leftOversize, rightOversize); - - if (rowSpacing !== null) - rowSpacing += oversize; - if (colSpacing !== null) - colSpacing += oversize; - - // Chrome highlights and window titles may exceed the workspace preview area - // and also the screen area if there is no overview element below/above/on_the_right of the workspace - // The original code tests whether window titles are out of the screen and applies correction accordingly - // That is a problem when workspaces are vertically stacked, because this method is called even during transitions between workspaces - // In V-Shell, this issue can be solved by reducing the workspace preview scale in the Settings - - // Original code - horizontal orientation only - /* if (containerBox) { - const monitor = Main.layoutManager.monitors[this._monitorIndex]; - - const bottomPoint = new Graphene.Point3D({ y: containerBox.y2 }); - const transformedBottomPoint = - this._container.apply_transform_to_point(bottomPoint); - const bottomFreeSpace = - (monitor.y + monitor.height) - transformedBottomPoint.y; - - const [, bottomOverlap] = window.overlapHeights(); - - if ((bottomOverlap + oversize) > bottomFreeSpace) - containerBox.y2 -= (bottomOverlap + oversize) - bottomFreeSpace; - }*/ - - // Alternative code reducing the box size unconditionally - /* if (containerBox) { - const [, bottomOverlap] = window.overlapHeights(); - - // Adjusting x1/x2 here is pointless, - // x1 only moves window previews to the right and down, x2 has no effect - // Prevent window previews from overlapping a workspace preview - oversize *= 1.5; - containerBox.y1 += oversize; - containerBox.y2 -= bottomOverlap + oversize; - }*/ - - return [rowSpacing, colSpacing, containerBox]; - }, - - _createBestLayout(area) { - const [rowSpacing, columnSpacing] = - this._adjustSpacingAndPadding(this._spacing, this._spacing, null); - - // We look for the largest scale that allows us to fit the - // largest row/tallest column on the workspace. - this._layoutStrategy = new UnalignedLayoutStrategy({ - monitor: Main.layoutManager.monitors[this._monitorIndex], - rowSpacing, - columnSpacing, - }); - - let lastLayout = null; - let lastNumColumns = -1; - let lastScale = 0; - let lastSpace = 0; - - for (let numRows = 1; ; numRows++) { - const numColumns = Math.ceil(this._sortedWindows.length / numRows); - - // If adding a new row does not change column count just stop - // (for instance: 9 windows, with 3 rows -> 3 columns, 4 rows -> - // 3 columns as well => just use 3 rows then) - if (numColumns === lastNumColumns) - break; - - const layout = this._layoutStrategy.computeLayout(this._sortedWindows, { - numRows, - }); - - const [scale, space] = this._layoutStrategy.computeScaleAndSpace(layout, area); - - if (lastLayout && !this._isBetterScaleAndSpace(lastScale, lastSpace, scale, space)) - break; - - lastLayout = layout; - lastNumColumns = numColumns; - lastScale = scale; - lastSpace = space; - } - - return lastLayout; - }, -}; - -class UnalignedLayoutStrategy extends Workspace.LayoutStrategy { - _newRow() { - // Row properties: - // - // * x, y are the position of row, relative to area - // - // * width, height are the scaled versions of fullWidth, fullHeight - // - // * width also has the spacing in between windows. It's not in - // fullWidth, as the spacing is constant, whereas fullWidth is - // meant to be scaled - // - // * neither height/fullHeight have any sort of spacing or padding - return { - x: 0, y: 0, - width: 0, height: 0, - fullWidth: 0, fullHeight: 0, - windows: [], - }; - } - - // Computes and returns an individual scaling factor for @window, - // to be applied in addition to the overall layout scale. - _computeWindowScale(window) { - // Since we align windows next to each other, the height of the - // thumbnails is much more important to preserve than the width of - // them, so two windows with equal height, but maybe differering - // widths line up. - let ratio = window.boundingBox.height / this._monitor.height; - - // The purpose of this manipulation here is to prevent windows - // from getting too small. For something like a calculator window, - // we need to bump up the size just a bit to make sure it looks - // good. We'll use a multiplier of 1.5 for this. - - // Map from [0, 1] to [1.5, 1] - return Util.lerp(1.5, 1, ratio); - } - - _computeRowSizes(layout) { - let { rows, scale } = layout; - for (let i = 0; i < rows.length; i++) { - let row = rows[i]; - row.width = row.fullWidth * scale + (row.windows.length - 1) * this._columnSpacing; - row.height = row.fullHeight * scale; - } - } - - _keepSameRow(row, window, width, idealRowWidth) { - if (row.fullWidth + width <= idealRowWidth) - return true; - - let oldRatio = row.fullWidth / idealRowWidth; - let newRatio = (row.fullWidth + width) / idealRowWidth; - - if (Math.abs(1 - newRatio) < Math.abs(1 - oldRatio)) - return true; - - return false; - } - - _sortRow(row) { - // Sort windows horizontally to minimize travel distance. - // This affects in what order the windows end up in a row. - row.windows.sort((a, b) => a.windowCenter.x - b.windowCenter.x); - } - - computeLayout(windows, layoutParams) { - layoutParams = Params.parse(layoutParams, { - numRows: 0, - }); - - if (layoutParams.numRows === 0) - throw new Error(`${this.constructor.name}: No numRows given in layout params`); - - const numRows = layoutParams.numRows; - - let rows = []; - let totalWidth = 0; - for (let i = 0; i < windows.length; i++) { - let window = windows[i]; - let s = this._computeWindowScale(window); - totalWidth += window.boundingBox.width * s; - } - - let idealRowWidth = totalWidth / numRows; - - // Sort windows vertically to minimize travel distance. - // This affects what rows the windows get placed in. - let sortedWindows = windows.slice(); - sortedWindows.sort((a, b) => a.windowCenter.y - b.windowCenter.y); - - let windowIdx = 0; - for (let i = 0; i < numRows; i++) { - let row = this._newRow(); - rows.push(row); - - for (; windowIdx < sortedWindows.length; windowIdx++) { - let window = sortedWindows[windowIdx]; - let s = this._computeWindowScale(window); - let width = window.boundingBox.width * s; - let height = window.boundingBox.height * s; - row.fullHeight = Math.max(row.fullHeight, height); - - // either new width is < idealWidth or new width is nearer from idealWidth then oldWidth - if (this._keepSameRow(row, window, width, idealRowWidth) || (i === numRows - 1)) { - row.windows.push(window); - row.fullWidth += width; - } else { - break; - } - } - } - - let gridHeight = 0; - let maxRow; - for (let i = 0; i < numRows; i++) { - let row = rows[i]; - this._sortRow(row); - - if (!maxRow || row.fullWidth > maxRow.fullWidth) - maxRow = row; - gridHeight += row.fullHeight; - } - - return { - numRows, - rows, - maxColumns: maxRow.windows.length, - gridWidth: maxRow.fullWidth, - gridHeight, - }; - } - - computeScaleAndSpace(layout, area) { - let hspacing = (layout.maxColumns - 1) * this._columnSpacing; - let vspacing = (layout.numRows - 1) * this._rowSpacing; - - let spacedWidth = area.width - hspacing; - let spacedHeight = area.height - vspacing; - - let horizontalScale = spacedWidth / layout.gridWidth; - let verticalScale = spacedHeight / layout.gridHeight; - - // Thumbnails should be less than 70% of the original size - let scale = Math.min( - horizontalScale, verticalScale, WINDOW_PREVIEW_MAXIMUM_SCALE); - - let scaledLayoutWidth = layout.gridWidth * scale + hspacing; - let scaledLayoutHeight = layout.gridHeight * scale + vspacing; - let space = (scaledLayoutWidth * scaledLayoutHeight) / (area.width * area.height); - - layout.scale = scale; - - return [scale, space]; - } - - computeWindowSlots(layout, area) { - this._computeRowSizes(layout); - - let { rows, scale } = layout; - - let slots = []; - - // Do this in three parts. - let heightWithoutSpacing = 0; - for (let i = 0; i < rows.length; i++) { - let row = rows[i]; - heightWithoutSpacing += row.height; - } - - let verticalSpacing = (rows.length - 1) * this._rowSpacing; - let additionalVerticalScale = Math.min(1, (area.height - verticalSpacing) / heightWithoutSpacing); - - // keep track how much smaller the grid becomes due to scaling - // so it can be centered again - let compensation = 0; - let y = 0; - - for (let i = 0; i < rows.length; i++) { - let row = rows[i]; - - // If this window layout row doesn't fit in the actual - // geometry, then apply an additional scale to it. - let horizontalSpacing = (row.windows.length - 1) * this._columnSpacing; - let widthWithoutSpacing = row.width - horizontalSpacing; - let additionalHorizontalScale = Math.min(1, (area.width - horizontalSpacing) / widthWithoutSpacing); - - if (additionalHorizontalScale < additionalVerticalScale) { - row.additionalScale = additionalHorizontalScale; - // Only consider the scaling in addition to the vertical scaling for centering. - compensation += (additionalVerticalScale - additionalHorizontalScale) * row.height; - } else { - row.additionalScale = additionalVerticalScale; - // No compensation when scaling vertically since centering based on a too large - // height would undo what vertical scaling is trying to achieve. - } - - row.x = area.x + (Math.max(area.width - (widthWithoutSpacing * row.additionalScale + horizontalSpacing), 0) / 2); - row.y = area.y + (Math.max(area.height - (heightWithoutSpacing + verticalSpacing), 0) / 2) + y; - y += row.height * row.additionalScale + this._rowSpacing; - } - - compensation /= 2; - - for (let i = 0; i < rows.length; i++) { - const row = rows[i]; - const rowY = row.y + compensation; - const rowHeight = row.height * row.additionalScale; - - let x = row.x; - for (let j = 0; j < row.windows.length; j++) { - let window = row.windows[j]; - - let s = scale * this._computeWindowScale(window) * row.additionalScale; - let cellWidth = window.boundingBox.width * s; - let cellHeight = window.boundingBox.height * s; - - s = Math.min(s, WINDOW_PREVIEW_MAXIMUM_SCALE); - let cloneWidth = window.boundingBox.width * s; - const cloneHeight = window.boundingBox.height * s; - - let cloneX = x + (cellWidth - cloneWidth) / 2; - let cloneY; - - // If there's only one row, align windows vertically centered inside the row - if (rows.length === 1) - cloneY = rowY + (rowHeight - cloneHeight) / 2; - // If there are multiple rows, align windows to the bottom edge of the row - else - cloneY = rowY + rowHeight - cellHeight; - - // Align with the pixel grid to prevent blurry windows at scale = 1 - cloneX = Math.floor(cloneX); - cloneY = Math.floor(cloneY); - - slots.push([cloneX, cloneY, cloneWidth, cloneHeight, window]); - x += cellWidth + this._columnSpacing; - } - } - return slots; - } -} - -const WorkspaceBackground = { - _updateBorderRadius(value = false) { - // don't round already rounded corners during exposing windows - if (value === false && opt.OVERVIEW_MODE === 1) - return; - - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - const cornerRadius = scaleFactor * opt.WS_PREVIEW_BG_RADIUS; - - const backgroundContent = this._bgManager.backgroundActor.content; - value = value !== false - ? value - : this._stateAdjustment.value; - - backgroundContent.rounded_clip_radius = - Util.lerp(0, cornerRadius, value); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/workspaceAnimation.js b/extensions/46/vertical-workspaces/lib/workspaceAnimation.js deleted file mode 100644 index e29e3ef..0000000 --- a/extensions/46/vertical-workspaces/lib/workspaceAnimation.js +++ /dev/null @@ -1,262 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * workspacesAnimation.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import GObject from 'gi://GObject'; -import St from 'gi://St'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Layout from 'resource:///org/gnome/shell/ui/layout.js'; -import * as WorkspaceSwitcherPopup from 'resource:///org/gnome/shell/ui/workspaceSwitcherPopup.js'; -import * as WorkspaceAnimation from 'resource:///org/gnome/shell/ui/workspaceAnimation.js'; -import * as Util from 'resource:///org/gnome/shell/misc/util.js'; - -let Me; -let opt; - -export const WorkspaceAnimationModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - this._origBaseDistance = null; - this._wsAnimationSwipeBeginId = 0; - this._wsAnimationSwipeUpdateId = 0; - this._wsAnimationSwipeEndId = 0; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('workspaceAnimationModule'); - const conflict = !WorkspaceAnimation.MonitorGroup; - if (conflict) - console.warn(`[${Me.metadata.name}] Warning: "WorkspaceAnimation" module disabled due to compatibility - GNOME Shell 45.1 or later is required`); - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WorkspaceAnimationModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('MonitorGroup', WorkspaceAnimation.MonitorGroup.prototype, MonitorGroup); - this._connectWsAnimationSwipeTracker(); - - console.debug(' WorkspaceAnimationModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - const reset = true; - this._connectWsAnimationSwipeTracker(reset); - - console.debug(' WorkspaceAnimationModule - Disabled'); - } - - _connectWsAnimationSwipeTracker(reset = false) { - if (reset) { - if (this._wsAnimationSwipeBeginId) { - Main.wm._workspaceAnimation._swipeTracker.disconnect(this._wsAnimationSwipeBeginId); - this._wsAnimationSwipeBeginId = 0; - } - if (this._wsAnimationSwipeEndId) { - Main.wm._workspaceAnimation._swipeTracker.disconnect(this._wsAnimationSwipeEndId); - this._wsAnimationSwipeEndId = 0; - } - } else if (!this._wsAnimationSwipeBeginId) { - // display ws switcher popup when gesture begins and connect progress - this._wsAnimationSwipeBeginId = Main.wm._workspaceAnimation._swipeTracker.connect('begin', () => this._connectWsAnimationProgress(true)); - // we want to be sure that popup with the final ws index show up when gesture ends - this._wsAnimationSwipeEndId = Main.wm._workspaceAnimation._swipeTracker.connect('end', (tracker, duration, endProgress) => this._connectWsAnimationProgress(false, endProgress)); - } - } - - _connectWsAnimationProgress(connect, endProgress = null) { - if (Main.overview.visible) - return; - - if (connect && !this._wsAnimationSwipeUpdateId) { - this._wsAnimationSwipeUpdateId = Main.wm._workspaceAnimation._swipeTracker.connect('update', (tracker, progress) => this._showWsSwitcherPopup(progress)); - } else if (!connect && this._wsAnimationSwipeUpdateId) { - Main.wm._workspaceAnimation._swipeTracker.disconnect(this._wsAnimationSwipeUpdateId); - this._wsAnimationSwipeUpdateId = 0; - this._showWsSwitcherPopup(Math.round(endProgress)); - } - } - - _showWsSwitcherPopup(progress) { - if (Main.overview.visible) - return; - - const wsIndex = Math.round(progress); - if (Main.wm._workspaceSwitcherPopup === null) { - Main.wm._workspaceSwitcherPopup = new WorkspaceSwitcherPopup.WorkspaceSwitcherPopup(); - Main.wm._workspaceSwitcherPopup.connect('destroy', () => { - Main.wm._workspaceSwitcherPopup = null; - }); - } - - Main.wm._workspaceSwitcherPopup.display(wsIndex); - } -}; - -const MonitorGroup = { - _init(monitor, workspaceIndices, movingWindow) { - St.Widget.prototype._init.bind(this)({ - clip_to_allocation: true, - style_class: 'workspace-animation', - }); - - this._monitor = monitor; - - const constraint = new Layout.MonitorConstraint({ index: monitor.index }); - this.add_constraint(constraint); - - this._container = new Clutter.Actor(); - this.add_child(this._container); - - const stickyGroup = new WorkspaceAnimation.WorkspaceGroup(null, monitor, movingWindow); - stickyGroup._windowRecords.forEach(r => { - const metaWin = r.windowActor.metaWindow; - // conky is sticky but should never get above other windows during ws animation - // so we hide it from the overlay group, we will see the original if not covered by other windows - if (metaWin.wm_class === 'conky') - r.clone.opacity = 0; - }); - this.add_child(stickyGroup); - - this._workspaceGroups = []; - - const workspaceManager = global.workspace_manager; - const vertical = workspaceManager.layout_rows === -1; - const activeWorkspace = workspaceManager.get_active_workspace(); - - let x = 0; - let y = 0; - - for (const i of workspaceIndices) { - const ws = workspaceManager.get_workspace_by_index(i); - const fullscreen = ws.list_windows().some(w => w.get_monitor() === monitor.index && w.is_fullscreen()); - - if (i > 0 && vertical && !fullscreen && monitor.index === Main.layoutManager.primaryIndex) { - // We have to shift windows up or down by the height of the panel to prevent having a - // visible gap between the windows while switching workspaces. Since fullscreen windows - // hide the panel, they don't need to be shifted up or down. - y -= Main.panel.height; - } - - const group = new WorkspaceAnimation.WorkspaceGroup(ws, monitor, movingWindow); - - this._workspaceGroups.push(group); - this._container.add_child(group); - group.set_position(x, y); - - if (vertical) - y += this.baseDistance; - else if (Clutter.get_default_text_direction() === Clutter.TextDirection.RTL) - x -= this.baseDistance; - else - x += this.baseDistance; - } - - this.progress = this.getWorkspaceProgress(activeWorkspace); - - if (monitor.index === Main.layoutManager.primaryIndex) { - this._workspacesAdjustment = Main.createWorkspacesAdjustment(this); - this.bind_property_full('progress', - this._workspacesAdjustment, 'value', - GObject.BindingFlags.SYNC_CREATE, - (bind, source) => { - const indices = [ - workspaceIndices[Math.floor(source)], - workspaceIndices[Math.ceil(source)], - ]; - return [true, Util.lerp(...indices, source % 1.0)]; - }, - null); - - this.connect('destroy', () => { - // for some reason _workspaceAdjustment bound to the progress property in V-Shell - // causes the adjustment doesn't reach a whole number - // when switching ws up and that breaks the showing overview animation - // as a workaround round workspacesDisplay._scrollAdjustment value on destroy - // but it should be handled elsewhere as this workaround doesn't work when this module is disabled - const workspacesAdj = Main.overview._overview.controls._workspacesDisplay._scrollAdjustment; - workspacesAdj.value = Math.round(workspacesAdj.value); - delete this._workspacesAdjustment; - }); - } - - if (!opt.STATIC_WS_SWITCHER_BG) - return; - - // we have two options to implement static bg feature - // one is adding background to monitorGroup - // but this one has disadvantage - sticky windows will be always on top of animated windows - // which is bad for conky, for example, that window should be always below - /* this._bgManager = new Background.BackgroundManager({ - container: this, - monitorIndex: this._monitor.index, - controlPosition: false, - });*/ - - // the second option is to make background of the monitorGroup transparent so the real desktop content will stay visible, - // hide windows that should be animated and keep only sticky windows - // we can keep certain sticky windows bellow and also extensions like DING (icons on desktop) will stay visible - this.set_style('background-color: transparent;'); - // stickyGroup holds the Always on Visible Workspace windows to keep them static and above other windows during animation - this._hiddenWindows = []; - // remove (hide) background wallpaper from the animation, we will see the original one - this._workspaceGroups.forEach(w => { - w._background.opacity = 0; - }); - // hide (scale to 0) all non-sticky windows, their clones will be animated - global.get_window_actors().forEach(actor => { - const metaWin = actor.metaWindow; - if (metaWin?.get_monitor() === this._monitor.index && - !(metaWin?.wm_class === 'conky' && metaWin?.is_on_all_workspaces()) && - !(metaWin?.wm_class === 'Gjs' && metaWin?.is_on_all_workspaces())) { // DING extension uses window with Gjs class - // hide original window. we cannot use opacity since it also affects clones. - // scaling them to 0 works well - actor.scale_x = 0; - this._hiddenWindows.push(actor); - } - }); - - // restore all hidden windows at the end of animation - // todo - actors removed during transition need to be removed from the list to avoid access to destroyed actor - this.connect('destroy', () => { - this._hiddenWindows.forEach(actor => { - actor.scale_x = 1; - }); - }); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/workspaceSwitcherPopup.js b/extensions/46/vertical-workspaces/lib/workspaceSwitcherPopup.js deleted file mode 100644 index cf3d4c1..0000000 --- a/extensions/46/vertical-workspaces/lib/workspaceSwitcherPopup.js +++ /dev/null @@ -1,107 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * workspacesSwitcherPopup.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as WorkspaceSwitcherPopup from 'resource:///org/gnome/shell/ui/workspaceSwitcherPopup.js'; - -let Me; -let opt; - -export const WorkspaceSwitcherPopupModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = opt.get('workspaceSwitcherPopupModule'); - const conflict = Me.Util.getEnabledExtensions('workspace-switcher-manager').length || - Me.Util.getEnabledExtensions('WsSwitcherPopupManager').length; - if (conflict && !reset) - console.warn(`[${Me.metadata.name}] Warning: "WorkspaceSwitcherPopup" module disabled due to potential conflict with another extension`); - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WorkspaceSwitcherPopupModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - this._overrides.addOverride('WorkspaceSwitcherPopup', WorkspaceSwitcherPopup.WorkspaceSwitcherPopup.prototype, WorkspaceSwitcherPopupCommon); - console.debug(' WorkspaceSwitcherPopupModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - console.debug(' WorkspaceSwitcherPopupModule - Disabled'); - } -}; - -const WorkspaceSwitcherPopupCommon = { - // injection to _init() - after__init() { - if (opt.ORIENTATION) { // 1-VERTICAL, 0-HORIZONTAL - this._list.vertical = true; - this._list.add_style_class_name('ws-switcher-vertical'); - } - this._list.set_style('margin: 0;'); - if (this.get_constraints()[0]) - this.remove_constraint(this.get_constraints()[0]); - }, - - // injection to display() - after_display() { - if (opt.WS_SW_POPUP_MODE) - this._setPopupPosition(); - else - this.opacity = 0; - }, - - _setPopupPosition() { - let workArea; - if (opt.WS_SW_POPUP_MODE === 1) - workArea = global.display.get_monitor_geometry(Main.layoutManager.primaryIndex); - else - workArea = global.display.get_monitor_geometry(global.display.get_current_monitor()); - - - let [, natHeight] = this.get_preferred_height(global.screen_width); - let [, natWidth] = this.get_preferred_width(natHeight); - let h = opt.WS_SW_POPUP_H_POSITION; - let v = opt.WS_SW_POPUP_V_POSITION; - this.x = workArea.x + Math.floor((workArea.width - natWidth) * h); - this.y = workArea.y + Math.floor((workArea.height - natHeight) * v); - this.set_position(this.x, this.y); - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/workspaceThumbnail.js b/extensions/46/vertical-workspaces/lib/workspaceThumbnail.js deleted file mode 100644 index cce5046..0000000 --- a/extensions/46/vertical-workspaces/lib/workspaceThumbnail.js +++ /dev/null @@ -1,1261 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * workspaceThumbnail.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import GLib from 'gi://GLib'; -import Clutter from 'gi://Clutter'; -import St from 'gi://St'; -import Meta from 'gi://Meta'; -import Shell from 'gi://Shell'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as DND from 'resource:///org/gnome/shell/ui/dnd.js'; -import * as AppDisplay from 'resource:///org/gnome/shell/ui/appDisplay.js'; -import * as OverviewControls from 'resource:///org/gnome/shell/ui/overviewControls.js'; -import * as WorkspaceThumbnail from 'resource:///org/gnome/shell/ui/workspaceThumbnail.js'; -import * as Background from 'resource:///org/gnome/shell/ui/background.js'; - -let Me; -let opt; - -const ThumbnailState = { - NEW: 0, - EXPANDING: 1, - EXPANDED: 2, - ANIMATING_IN: 3, - NORMAL: 4, - REMOVING: 5, - ANIMATING_OUT: 6, - ANIMATED_OUT: 7, - COLLAPSING: 8, - DESTROYED: 9, -}; - -const ControlsState = OverviewControls.ControlsState; - -const WORKSPACE_CUT_SCALE = 0.15; -const WORKSPACE_KEEP_ALIVE_TIME = 100; - -export const WorkspaceThumbnailModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = true; - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WorkspaceThumbnailModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - // don't limit max thumbnail scale for other clients than overview, specifically AATWS. - // this variable is not yet implemented in 45.beta.1 - - this._overrides.addOverride('WorkspaceThumbnail', WorkspaceThumbnail.WorkspaceThumbnail.prototype, WorkspaceThumbnailCommon); - this._overrides.addOverride('ThumbnailsBoxCommon', WorkspaceThumbnail.ThumbnailsBox.prototype, ThumbnailsBoxCommon); - this._overrides.addOverride('WindowClone', WorkspaceThumbnail.WindowClone.prototype, WindowClone); - - // replacing opt.ORIENTATION local constant with boxOrientation internal variable allows external customers such as the AATWS extension to control the box orientation. - Main.overview._overview.controls._thumbnailsBox._boxOrientation = opt.ORIENTATION; - - console.debug(' WorkspaceThumbnailModule - Activated'); - } - - _disableModule() { - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - console.debug(' WorkspaceThumbnailModule - Disabled'); - } -}; - -const WorkspaceThumbnailCommon = { - // injection to _init() - after__init() { - // layout manager allows aligning widget children - this.layout_manager = new Clutter.BinLayout(); - // adding layout manager to tmb widget breaks wallpaper background aligning and rounded corners - // unless border is removed - if (opt.SHOW_WS_TMB_BG) - this.add_style_class_name('ws-tmb-labeled'); - else - this.add_style_class_name('ws-tmb-transparent'); - - // add workspace thumbnails labels if enabled - if (opt.SHOW_WST_LABELS) { // 0 - disable - const getLabel = function () { - const wsIndex = this.metaWorkspace.index(); - let label = `${wsIndex + 1}`; - if (opt.SHOW_WST_LABELS === 2) { // 2 - index + workspace name - const settings = Me.getSettings('org.gnome.desktop.wm.preferences'); - const wsLabels = settings.get_strv('workspace-names'); - if (wsLabels.length > wsIndex && wsLabels[wsIndex]) - label += `: ${wsLabels[wsIndex]}`; - } else if (opt.SHOW_WST_LABELS === 3) { // 3- index + app name - // global.display.get_tab_list offers workspace filtering using the second argument, but... - // ... it sometimes includes windows from other workspaces, like minimized VBox machines, after Shell restarts - const metaWin = global.display.get_tab_list(0, null).filter( - w => w.get_monitor() === this.monitorIndex && w.get_workspace().index() === wsIndex)[0]; - - if (metaWin) { - const tracker = Shell.WindowTracker.get_default(); - const app = tracker.get_window_app(metaWin); - label += `: ${app ? app.get_name() : ''}`; - } - } else if (opt.SHOW_WST_LABELS === 4) { - const metaWin = global.display.get_tab_list(0, null).filter( - w => w.get_monitor() === this.monitorIndex && w.get_workspace().index() === wsIndex)[0]; - - if (metaWin) - label += `: ${metaWin.title}`; - } - return label; - }.bind(this); - - const label = getLabel(); - - this._wsLabel = new St.Label({ - text: label, - style_class: 'ws-tmb-label', - x_align: Clutter.ActorAlign.FILL, - y_align: Clutter.ActorAlign.END, - x_expand: true, - y_expand: true, - }); - - this._wsLabel._maxOpacity = 255; - this._wsLabel.opacity = this._wsLabel._maxOpacity; - - this.add_child(this._wsLabel); - this.set_child_above_sibling(this._wsLabel, null); - - this._wsIndexConId = this.metaWorkspace.connect('notify::workspace-index', () => { - const newLabel = getLabel(); - this._wsLabel.text = newLabel; - // avoid possibility of accessing non existing ws - if (this._updateLabelTimeout) { - GLib.source_remove(this._updateLabelTimeout); - this._updateLabelTimeout = 0; - } - }); - this._nWindowsConId = this.metaWorkspace.connect('notify::n-windows', () => { - if (this._updateLabelTimeout) - return; - // wait for new data - this._updateLabelTimeout = GLib.timeout_add(GLib.PRIORITY_DEFAULT, 250, () => { - const newLabel = getLabel(); - this._wsLabel.text = newLabel; - this._updateLabelTimeout = 0; - return GLib.SOURCE_REMOVE; - }); - }); - } - - if (opt.CLOSE_WS_BUTTON_MODE) { - const closeButton = new St.Icon({ - style_class: 'workspace-close-button', - icon_name: 'window-close-symbolic', - x_align: Clutter.ActorAlign.END, - y_align: Clutter.ActorAlign.START, - x_expand: true, - y_expand: true, - reactive: true, - opacity: 0, - }); - - closeButton.connect('button-release-event', () => { - if (opt.CLOSE_WS_BUTTON_MODE) { - this._closeWorkspace(); - return Clutter.EVENT_STOP; - } else { - return Clutter.EVENT_PROPAGATE; - } - }); - - closeButton.connect('button-press-event', () => { - return Clutter.EVENT_STOP; - }); - - closeButton.connect('enter-event', () => { - closeButton.opacity = 255; - if (!Meta.prefs_get_dynamic_workspaces() || (Meta.prefs_get_dynamic_workspaces() && global.workspace_manager.get_n_workspaces() - 1 !== this.metaWorkspace.index())) { - // color the button red if ready to react on clicks - if (opt.CLOSE_WS_BUTTON_MODE < 3 || (opt.CLOSE_WS_BUTTON_MODE === 3 && Me.Util.isCtrlPressed())) - closeButton.add_style_class_name('workspace-close-button-hover'); - } - }); - - closeButton.connect('leave-event', () => { - closeButton.remove_style_class_name('workspace-close-button-hover'); - }); - - this.add_child(closeButton); - this._closeButton = closeButton; - - this.reactive = true; - this._lastCloseClickTime = 0; - } - - if (opt.SHOW_WST_LABELS_ON_HOVER) - this._wsLabel.opacity = 0; - - this.connect('enter-event', () => { - if (opt.CLOSE_WS_BUTTON_MODE && (!Meta.prefs_get_dynamic_workspaces() || (Meta.prefs_get_dynamic_workspaces() && global.workspace_manager.get_n_workspaces() - 1 !== this.metaWorkspace.index()))) - this._closeButton.opacity = 200; - if (opt.SHOW_WST_LABELS_ON_HOVER) { - this._wsLabel.ease({ - duration: 100, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - opacity: this._wsLabel._maxOpacity, - }); - } - }); - - this.connect('leave-event', () => { - this._closeButton.opacity = 0; - if (opt.SHOW_WST_LABELS_ON_HOVER) { - this._wsLabel.ease({ - duration: 100, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - opacity: 0, - }); - } - }); - - if (opt.SHOW_WS_TMB_BG) { - this._bgManager = new Background.BackgroundManager({ - monitorIndex: this.monitorIndex, - container: this._viewport, - vignette: false, - controlPosition: false, - }); - - this._viewport.set_child_below_sibling(this._bgManager.backgroundActor, null); - - // full brightness of the thumbnail bg draws unnecessary attention - // there is a grey bg under the wallpaper - this._bgManager.backgroundActor.opacity = 220; - } - - this.connect('destroy', () => { - if (this._wsIndexConId) - this.metaWorkspace.disconnect(this._wsIndexConId); - - if (this._nWindowsConId) - this.metaWorkspace.disconnect(this._nWindowsConId); - - if (this._updateLabelTimeout) - GLib.source_remove(this._updateLabelTimeout); - - if (this._bgManager) - this._bgManager.destroy(); - }); - }, - - _closeWorkspace() { - // CLOSE_WS_BUTTON_MODE 1: single click, 2: double-click, 3: Ctrl - - if (opt.CLOSE_WS_BUTTON_MODE === 2) { - const doubleClickTime = Clutter.Settings.get_default().double_click_time; - const clickDelay = Date.now() - this._lastCloseClickTime; - if (clickDelay > doubleClickTime) { - this._lastCloseClickTime = Date.now(); - return; - } - } else if (opt.CLOSE_WS_BUTTON_MODE === 3 && !Me.Util.isCtrlPressed()) { - return; - } - - // close windows on this monitor - const windows = global.display.get_tab_list(0, null).filter( - w => w.get_monitor() === this.monitorIndex && w.get_workspace() === this.metaWorkspace - ); - - for (let i = 0; i < windows.length; i++) { - if (!windows[i].is_on_all_workspaces()) - windows[i].delete(global.get_current_time() + i); - } - }, - - activate(time) { - if (this.state > ThumbnailState.NORMAL) - return; - - // if Static Workspace overview mode active, a click on the already active workspace should activate the window picker mode - const wsIndex = this.metaWorkspace.index(); - const lastWsIndex = global.display.get_workspace_manager().get_n_workspaces() - 1; - const stateAdjustment = Main.overview._overview.controls._stateAdjustment; - - if (stateAdjustment.value === ControlsState.APP_GRID) { - if (this.metaWorkspace.active) { - Main.overview._overview.controls._shiftState(Meta.MotionDirection.DOWN); - // if searchActive, hide it immediately - Main.overview.searchEntry.set_text(''); - } else { - this.metaWorkspace.activate(time); - } - } else if (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE && wsIndex < lastWsIndex) { - if (stateAdjustment.value > 1) - stateAdjustment.value = 1; - - - // spread windows - // in OVERVIEW MODE 2 windows are not spread and workspace is not scaled - // we need to repeat transition to the overview state 1 (window picker), but with spreading windows animation - if (this.metaWorkspace.active) { - Main.overview.searchController._setSearchActive(false); - opt.WORKSPACE_MODE = 1; - // setting value to 0 would reset WORKSPACE_MODE - stateAdjustment.value = 0.01; - stateAdjustment.ease(1, { - duration: 200, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } else { - // switch ws - this.metaWorkspace.activate(time); - } - // a click on the current workspace should go back to the main view - } else if (this.metaWorkspace.active) { - Main.overview.hide(); - } else { - this.metaWorkspace.activate(time); - } - }, - - // Draggable target interface used only by ThumbnailsBox - handleDragOverInternal(source, actor, time) { - if (source === Main.xdndHandler) { - this.metaWorkspace.activate(time); - return DND.DragMotionResult.CONTINUE; - } - - if (this.state > ThumbnailState.NORMAL) - return DND.DragMotionResult.CONTINUE; - - if (source.metaWindow && - !this._isMyWindow(source.metaWindow.get_compositor_private())) - return DND.DragMotionResult.MOVE_DROP; - if (source.app && source.app.can_open_new_window()) - return DND.DragMotionResult.COPY_DROP; - if (!source.app && source.shellWorkspaceLaunch) - return DND.DragMotionResult.COPY_DROP; - - if (source instanceof AppDisplay.FolderIcon) - return DND.DragMotionResult.COPY_DROP; - - - return DND.DragMotionResult.CONTINUE; - }, - - acceptDropInternal(source, actor, time) { - if (this.state > ThumbnailState.NORMAL) - return false; - - if (source.metaWindow) { - let win = source.metaWindow.get_compositor_private(); - if (this._isMyWindow(win)) - return false; - - let metaWindow = win.get_meta_window(); - Main.moveWindowToMonitorAndWorkspace(metaWindow, - this.monitorIndex, this.metaWorkspace.index()); - return true; - } else if (source.app && source.app.can_open_new_window()) { - if (source.animateLaunchAtPos) - source.animateLaunchAtPos(actor.x, actor.y); - - source.app.open_new_window(this.metaWorkspace.index()); - return true; - } else if (!source.app && source.shellWorkspaceLaunch) { - // While unused in our own drag sources, shellWorkspaceLaunch allows - // extensions to define custom actions for their drag sources. - source.shellWorkspaceLaunch({ - workspace: this.metaWorkspace.index(), - timestamp: time, - }); - return true; - } else if (source instanceof AppDisplay.FolderIcon) { - for (let app of source.view._apps) { - // const app = Shell.AppSystem.get_default().lookup_app(id); - app.open_new_window(this.metaWorkspace.index()); - } - } - - return false; - }, -}; - -const ThumbnailsBoxCommon = { - after__init(scrollAdjustment, monitorIndex, orientation = opt.ORIENTATION) { - this._boxOrientation = orientation; - }, - - _activateThumbnailAtPoint(stageX, stageY, time, activateCurrent = false) { - if (activateCurrent) { - const thumbnail = this._thumbnails.find(t => t.metaWorkspace.active); - if (thumbnail) - thumbnail.activate(time); - return; - } - const [r_, x, y] = this.transform_stage_point(stageX, stageY); - - let thumbnail; - - if (this._boxOrientation) - thumbnail = this._thumbnails.find(t => y >= t.y && y <= t.y + t.height); - else - thumbnail = this._thumbnails.find(t => x >= t.x && x <= t.x + t.width); - - if (thumbnail) - thumbnail.activate(time); - }, - - acceptDrop(source, actor, x, y, time) { - if (this._dropWorkspace !== -1) { - return this._thumbnails[this._dropWorkspace].acceptDropInternal(source, actor, time); - } else if (this._dropPlaceholderPos !== -1) { - if (!source.metaWindow && - (!source.app || !source.app.can_open_new_window()) && - (source.app || !source.shellWorkspaceLaunch) && - !(source instanceof AppDisplay.FolderIcon)) - return false; - - - let isWindow = !!source.metaWindow; - - let newWorkspaceIndex; - [newWorkspaceIndex, this._dropPlaceholderPos] = [this._dropPlaceholderPos, -1]; - this._spliceIndex = newWorkspaceIndex; - - Main.wm.insertWorkspace(newWorkspaceIndex); - - if (isWindow) { - // Move the window to our monitor first if necessary. - let thumbMonitor = this._thumbnails[newWorkspaceIndex].monitorIndex; - Main.moveWindowToMonitorAndWorkspace(source.metaWindow, - thumbMonitor, newWorkspaceIndex, true); - } else if (source.app && source.app.can_open_new_window()) { - if (source.animateLaunchAtPos) - source.animateLaunchAtPos(actor.x, actor.y); - - source.app.open_new_window(newWorkspaceIndex); - } else if (!source.app && source.shellWorkspaceLaunch) { - // While unused in our own drag sources, shellWorkspaceLaunch allows - // extensions to define custom actions for their drag sources. - source.shellWorkspaceLaunch({ - workspace: newWorkspaceIndex, - timestamp: time, - }); - } else if (source instanceof AppDisplay.FolderIcon) { - for (let app of source.view._apps) { - // const app = Shell.AppSystem.get_default().lookup_app(id); - app.open_new_window(newWorkspaceIndex); - } - } - - if (source.app || (!source.app && source.shellWorkspaceLaunch)) { - // This new workspace will be automatically removed if the application fails - // to open its first window within some time, as tracked by Shell.WindowTracker. - // Here, we only add a very brief timeout to avoid the _immediate_ removal of the - // workspace while we wait for the startup sequence to load. - let workspaceManager = global.workspace_manager; - Main.wm.keepWorkspaceAlive(workspaceManager.get_workspace_by_index(newWorkspaceIndex), - WORKSPACE_KEEP_ALIVE_TIME); - } - - // Start the animation on the workspace (which is actually - // an old one which just became empty) - let thumbnail = this._thumbnails[newWorkspaceIndex]; - this._setThumbnailState(thumbnail, ThumbnailState.NEW); - thumbnail.slide_position = 1; - thumbnail.collapse_fraction = 1; - - this._queueUpdateStates(); - - return true; - } else { - return false; - } - }, - - handleDragOver(source, actor, x, y, time) { - // switch axis for vertical orientation - if (this._boxOrientation) - x = y; - - if (!source.metaWindow && - (!source.app || !source.app.can_open_new_window()) && - (source.app || !source.shellWorkspaceLaunch) && - source !== Main.xdndHandler && !(source instanceof AppDisplay.FolderIcon)) - return DND.DragMotionResult.CONTINUE; - - const rtl = Clutter.get_default_text_direction() === Clutter.TextDirection.RTL; - let canCreateWorkspaces = Meta.prefs_get_dynamic_workspaces(); - let spacing = this.get_theme_node().get_length('spacing'); - - this._dropWorkspace = -1; - let placeholderPos = -1; - let length = this._thumbnails.length; - for (let i = 0; i < length; i++) { - const index = rtl ? length - i - 1 : i; - - if (canCreateWorkspaces && source !== Main.xdndHandler) { - const [targetStart, targetEnd] = - this._getPlaceholderTarget(index, spacing, rtl); - - if (x > targetStart && x <= targetEnd) { - placeholderPos = index; - break; - } - } - - if (this._withinWorkspace(x, index, rtl)) { - this._dropWorkspace = index; - break; - } - } - - if (this._dropPlaceholderPos !== placeholderPos) { - this._dropPlaceholderPos = placeholderPos; - this.queue_relayout(); - } - - if (this._dropWorkspace !== -1) - return this._thumbnails[this._dropWorkspace].handleDragOverInternal(source, actor, time); - else if (this._dropPlaceholderPos !== -1) - return source.metaWindow ? DND.DragMotionResult.MOVE_DROP : DND.DragMotionResult.COPY_DROP; - else - return DND.DragMotionResult.CONTINUE; - }, - - _updateStates() { - const controlsManager = Main.overview._overview.controls; - const { currentState } = controlsManager._stateAdjustment.getStateTransitionParams(); - this.SLIDE_ANIMATION_TIME = 200; - this.RESCALE_ANIMATION_TIME = 200; - // remove rescale animation during this scale transition, it is redundant and delayed - if ((currentState < 2 && currentState > 1) || controlsManager._searchController.searchActive) - this.RESCALE_ANIMATION_TIME = 0; - - this._updateStateId = 0; - - // If we are animating the indicator, wait - if (this._animatingIndicator) - return; - - // Likewise if we are in the process of hiding - if (!this._shouldShow && this.visible) - return; - - // Then slide out any thumbnails that have been destroyed - this._iterateStateThumbnails(ThumbnailState.REMOVING, thumbnail => { - this._setThumbnailState(thumbnail, ThumbnailState.ANIMATING_OUT); - - thumbnail.ease_property('slide-position', 1, { - duration: this.SLIDE_ANIMATION_TIME, - mode: Clutter.AnimationMode.LINEAR, - onComplete: () => { - this._setThumbnailState(thumbnail, ThumbnailState.ANIMATED_OUT); - this._queueUpdateStates(); - }, - }); - }); - - // As long as things are sliding out, don't proceed - if (this._stateCounts[ThumbnailState.ANIMATING_OUT] > 0) - return; - - // Once that's complete, we can start scaling to the new size, - // collapse any removed thumbnails and expand added ones - this._iterateStateThumbnails(ThumbnailState.ANIMATED_OUT, thumbnail => { - this._setThumbnailState(thumbnail, ThumbnailState.COLLAPSING); - thumbnail.ease_property('collapse-fraction', 1, { - duration: this.RESCALE_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._stateCounts[thumbnail.state]--; - thumbnail.state = ThumbnailState.DESTROYED; - - let index = this._thumbnails.indexOf(thumbnail); - this._thumbnails.splice(index, 1); - thumbnail.destroy(); - - this._queueUpdateStates(); - }, - }); - }); - - this._iterateStateThumbnails(ThumbnailState.NEW, thumbnail => { - this._setThumbnailState(thumbnail, ThumbnailState.EXPANDING); - thumbnail.ease_property('collapse-fraction', 0, { - duration: this.SLIDE_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._setThumbnailState(thumbnail, ThumbnailState.EXPANDED); - this._queueUpdateStates(); - }, - }); - }); - - if (this._pendingScaleUpdate) { - this.ease_property('scale', this._targetScale, { - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - duration: this.RESCALE_ANIMATION_TIME, - onComplete: () => this._queueUpdateStates(), - }); - this._queueUpdateStates(); - this._pendingScaleUpdate = false; - } - - // Wait until that's done - if (this._scale !== this._targetScale || - this._stateCounts[ThumbnailState.COLLAPSING] > 0 || - this._stateCounts[ThumbnailState.EXPANDING] > 0) - return; - - // And then slide in any new thumbnails - this._iterateStateThumbnails(ThumbnailState.EXPANDED, thumbnail => { - this._setThumbnailState(thumbnail, ThumbnailState.ANIMATING_IN); - thumbnail.ease_property('slide-position', 0, { - duration: this.SLIDE_ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._setThumbnailState(thumbnail, ThumbnailState.NORMAL); - }, - }); - }); - }, - - _getPlaceholderTarget(...args) { - if (this._boxOrientation) - return ThumbnailsBoxVertical._getPlaceholderTarget.bind(this)(...args); - else - return ThumbnailsBoxHorizontal._getPlaceholderTarget.bind(this)(...args); - }, - - _withinWorkspace(...args) { - if (this._boxOrientation) - return ThumbnailsBoxVertical._withinWorkspace.bind(this)(...args); - else - return ThumbnailsBoxHorizontal._withinWorkspace.bind(this)(...args); - }, - - vfunc_get_preferred_width(...args) { - if (this._boxOrientation) - return ThumbnailsBoxVertical.vfunc_get_preferred_width.bind(this)(...args); - else - return ThumbnailsBoxHorizontal.vfunc_get_preferred_width.bind(this)(...args); - }, - - vfunc_get_preferred_height(...args) { - if (this._boxOrientation) - return ThumbnailsBoxVertical.vfunc_get_preferred_height.bind(this)(...args); - else - return ThumbnailsBoxHorizontal.vfunc_get_preferred_height.bind(this)(...args); - }, - - vfunc_allocate(...args) { - if (this._boxOrientation) - return ThumbnailsBoxVertical.vfunc_allocate.bind(this)(...args); - else - return ThumbnailsBoxHorizontal.vfunc_allocate.bind(this)(...args); - }, - - _updateShouldShow(...args) { - if (this._boxOrientation) - return ThumbnailsBoxVertical._updateShouldShow.bind(this)(...args); - else - return ThumbnailsBoxHorizontal._updateShouldShow.bind(this)(...args); - }, -}; - -function _getWorkspaceCutSize(tmbSize, index) { - let cutSize = WORKSPACE_CUT_SCALE * tmbSize; - // Compensate for the missing thumbnail in front of the first one - if (index === 0) - cutSize *= 1.5; - return Math.floor(cutSize); -} - -const ThumbnailsBoxVertical = { - _getPlaceholderTarget(index, spacing, rtl) { - this._dropPlaceholder.add_style_class_name('placeholder-vertical'); - const workspace = this._thumbnails[index]; - const WORKSPACE_CUT_SIZE = _getWorkspaceCutSize(workspace.height, index); - - let targetY1; - let targetY2; - - if (rtl) { - const baseY = workspace.y + workspace.height; - targetY1 = baseY - WORKSPACE_CUT_SIZE; - targetY2 = baseY + spacing + WORKSPACE_CUT_SIZE; - } else { - targetY1 = workspace.y - spacing - WORKSPACE_CUT_SIZE; - targetY2 = workspace.y + WORKSPACE_CUT_SIZE; - } - - if (index === 0) { - if (rtl) - targetY2 -= spacing + WORKSPACE_CUT_SIZE; - else - targetY1 += spacing + WORKSPACE_CUT_SIZE; - } - - if (index === this._dropPlaceholderPos) { - const placeholderHeight = this._dropPlaceholder.get_height() + spacing; - if (rtl) - targetY2 += placeholderHeight; - else - targetY1 -= placeholderHeight; - } - - return [targetY1, targetY2]; - }, - - _withinWorkspace(y, index, rtl) { - const length = this._thumbnails.length; - const workspace = this._thumbnails[index]; - const WORKSPACE_CUT_SIZE = _getWorkspaceCutSize(workspace.height, index); - - let workspaceY1 = workspace.y + WORKSPACE_CUT_SIZE; - let workspaceY2 = workspace.y + workspace.height - WORKSPACE_CUT_SIZE; - - if (index === length - 1) { - if (rtl) - workspaceY1 -= WORKSPACE_CUT_SIZE; - else - workspaceY2 += WORKSPACE_CUT_SIZE; - } - - return y > workspaceY1 && y <= workspaceY2; - }, - - vfunc_get_preferred_width(forHeight) { - if (forHeight < 10) - return [this._porthole.width, this._porthole.width]; - - let themeNode = this.get_theme_node(); - - forHeight = themeNode.adjust_for_width(forHeight); - - let spacing = themeNode.get_length('spacing'); - let nWorkspaces = this._thumbnails.length; - let totalSpacing = (nWorkspaces - 1) * spacing; - - const avail = forHeight - totalSpacing; - - let scale = (avail / nWorkspaces) / this._porthole.height; - - const width = Math.round(this._porthole.width * scale); - return themeNode.adjust_preferred_height(width, width); - }, - - vfunc_get_preferred_height(forWidth) { - if (forWidth < 10) - return [0, this._porthole.height]; - let themeNode = this.get_theme_node(); - - let spacing = themeNode.get_length('spacing'); - let nWorkspaces = this._thumbnails.length; - - // remove also top/bottom box padding - let totalSpacing = (nWorkspaces - 3) * spacing; - - const ratio = this._porthole.width / this._porthole.height; - const tmbHeight = themeNode.adjust_for_width(forWidth) / ratio; - - const naturalHeight = Math.round( - this._thumbnails.reduce((accumulator, thumbnail/* , index*/) => { - const progress = 1 - thumbnail.collapse_fraction; - const height = tmbHeight * progress; - return accumulator + height; - }, 0) - ); - return themeNode.adjust_preferred_width(totalSpacing, naturalHeight); - }, - - // removes extra space (extraWidth in the original function), we need the box as accurate as possible - // for precise app grid transition animation - vfunc_allocate(box) { - this.set_allocation(box); - - let rtl = Clutter.get_default_text_direction() === Clutter.TextDirection.RTL; - - if (this._thumbnails.length === 0) // not visible - return; - - let themeNode = this.get_theme_node(); - box = themeNode.get_content_box(box); - - const portholeWidth = this._porthole.width; - const portholeHeight = this._porthole.height; - const spacing = themeNode.get_length('spacing'); - - /* const nWorkspaces = this._thumbnails.length;*/ - - // Compute the scale we'll need once everything is updated, - // unless we are currently transitioning - if (this._expandFraction === 1) { - // remove size "breathing" during adding/removing workspaces - - /* const totalSpacing = (nWorkspaces - 1) * spacing; - const availableHeight = (box.get_height() - totalSpacing) / nWorkspaces; */ - - const hScale = box.get_width() / portholeWidth; - /* const vScale = availableHeight / portholeHeight;*/ - const vScale = box.get_height() / portholeHeight; - const newScale = Math.min(hScale, vScale); - - if (newScale !== this._targetScale) { - if (this._targetScale > 0) { - // We don't ease immediately because we need to observe the - // ordering in queueUpdateStates - if workspaces have been - // removed we need to slide them out as the first thing. - this._targetScale = newScale; - this._pendingScaleUpdate = true; - } else { - this._targetScale = this._scale = newScale; - } - - this._queueUpdateStates(); - } - } - - const ratio = portholeWidth / portholeHeight; - const thumbnailFullHeight = Math.round(portholeHeight * this._scale); - const thumbnailWidth = Math.round(thumbnailFullHeight * ratio); - const thumbnailHeight = thumbnailFullHeight * this._expandFraction; - const roundedVScale = thumbnailHeight / portholeHeight; - - let indicatorValue = this._scrollAdjustment.value; - let indicatorUpperWs = Math.ceil(indicatorValue); - let indicatorLowerWs = Math.floor(indicatorValue); - - let indicatorLowerY1 = 0; - let indicatorLowerY2 = 0; - let indicatorUpperY1 = 0; - let indicatorUpperY2 = 0; - - let indicatorThemeNode = this._indicator.get_theme_node(); - let indicatorTopFullBorder = indicatorThemeNode.get_padding(St.Side.TOP) + indicatorThemeNode.get_border_width(St.Side.TOP); - let indicatorBottomFullBorder = indicatorThemeNode.get_padding(St.Side.BOTTOM) + indicatorThemeNode.get_border_width(St.Side.BOTTOM); - let indicatorLeftFullBorder = indicatorThemeNode.get_padding(St.Side.LEFT) + indicatorThemeNode.get_border_width(St.Side.LEFT); - let indicatorRightFullBorder = indicatorThemeNode.get_padding(St.Side.RIGHT) + indicatorThemeNode.get_border_width(St.Side.RIGHT); - - let y = box.y1; - - if (this._dropPlaceholderPos === -1) { - this._dropPlaceholder.allocate_preferred_size( - ...this._dropPlaceholder.get_position()); - - const laters = global.compositor.get_laters(); - laters.add(Meta.LaterType.BEFORE_REDRAW, () => { - this._dropPlaceholder.hide(); - }); - } - - let childBox = new Clutter.ActorBox(); - - for (let i = 0; i < this._thumbnails.length; i++) { - const thumbnail = this._thumbnails[i]; - if (i > 0) - y += spacing - Math.round(thumbnail.collapse_fraction * spacing); - - const x1 = box.x1; - const x2 = x1 + thumbnailWidth; - - if (i === this._dropPlaceholderPos) { - let [, placeholderHeight] = this._dropPlaceholder.get_preferred_width(-1); - childBox.x1 = x1; - childBox.x2 = x2; - - if (rtl) { - childBox.y2 = box.y2 - Math.round(y); - childBox.y1 = box.y2 - Math.round(y + placeholderHeight); - } else { - childBox.y1 = Math.round(y); - childBox.y2 = Math.round(y + placeholderHeight); - } - - this._dropPlaceholder.allocate(childBox); - - const laters = global.compositor.get_laters(); - laters.add(Meta.LaterType.BEFORE_REDRAW, () => { - this._dropPlaceholder.show(); - }); - y += placeholderHeight + spacing; - } - - // We might end up with thumbnailWidth being something like 99.33 - // pixels. To make this work and not end up with a gap at the end, - // we need some thumbnails to be 99 pixels and some 100 pixels width; - // we compute an actual scale separately for each thumbnail. - const y1 = Math.round(y); - const y2 = Math.round(y + thumbnailHeight); - const roundedHScale = (y2 - y1) / portholeHeight; - - // Allocating a scaled actor is funny - x1/y1 correspond to the origin - // of the actor, but x2/y2 are increased by the *unscaled* size. - if (rtl) { - childBox.y2 = box.y2 - y1; - childBox.y1 = box.y2 - (y1 + thumbnailHeight); - } else { - childBox.y1 = y1; - childBox.y2 = y1 + thumbnailHeight; - } - childBox.x1 = x1; - childBox.x2 = x1 + thumbnailWidth; - - thumbnail.setScale(roundedHScale, roundedVScale); - thumbnail.allocate(childBox); - - if (i === indicatorUpperWs) { - indicatorUpperY1 = childBox.y1; - indicatorUpperY2 = childBox.y2; - } - if (i === indicatorLowerWs) { - indicatorLowerY1 = childBox.y1; - indicatorLowerY2 = childBox.y2; - } - - // We round the collapsing portion so that we don't get thumbnails resizing - // during an animation due to differences in rounded, but leave the uncollapsed - // portion unrounded so that non-animating we end up with the right total - y += thumbnailHeight - Math.round(thumbnailHeight * thumbnail.collapse_fraction); - } - - childBox.x1 = box.x1; - childBox.x2 = box.x1 + thumbnailWidth; - - const indicatorY1 = indicatorLowerY1 + - (indicatorUpperY1 - indicatorLowerY1) * (indicatorValue % 1); - const indicatorY2 = indicatorLowerY2 + - (indicatorUpperY2 - indicatorLowerY2) * (indicatorValue % 1); - - childBox.y1 = indicatorY1 - indicatorTopFullBorder; - childBox.y2 = indicatorY2 + indicatorBottomFullBorder; - childBox.x1 -= indicatorLeftFullBorder; - childBox.x2 += indicatorRightFullBorder; - this._indicator.allocate(childBox); - }, - - _updateShouldShow() { - const shouldShow = opt.SHOW_WS_TMB; - if (this._shouldShow === shouldShow) - return; - - this._shouldShow = shouldShow; - this.notify('should-show'); - }, -}; - -// ThumbnailsBox Horizontal - -const ThumbnailsBoxHorizontal = { - _getPlaceholderTarget(index, spacing, rtl) { - const workspace = this._thumbnails[index]; - const WORKSPACE_CUT_SIZE = _getWorkspaceCutSize(workspace.width, index); - - let targetX1; - let targetX2; - - if (rtl) { - const baseX = workspace.x + workspace.width; - targetX1 = baseX - WORKSPACE_CUT_SIZE; - targetX2 = baseX + spacing + WORKSPACE_CUT_SIZE; - } else { - targetX1 = workspace.x - spacing - WORKSPACE_CUT_SIZE; - targetX2 = workspace.x + WORKSPACE_CUT_SIZE; - } - - if (index === 0) { - if (rtl) - targetX2 -= spacing + WORKSPACE_CUT_SIZE; - else - targetX1 += spacing + WORKSPACE_CUT_SIZE; - } - - if (index === this._dropPlaceholderPos) { - const placeholderWidth = this._dropPlaceholder.get_width() + spacing; - if (rtl) - targetX2 += placeholderWidth; - else - targetX1 -= placeholderWidth; - } - - return [targetX1, targetX2]; - }, - - _withinWorkspace(x, index, rtl) { - const length = this._thumbnails.length; - const workspace = this._thumbnails[index]; - const WORKSPACE_CUT_SIZE = _getWorkspaceCutSize(workspace.width, index); - - let workspaceX1 = workspace.x + WORKSPACE_CUT_SIZE; - let workspaceX2 = workspace.x + workspace.width - WORKSPACE_CUT_SIZE; - - if (index === length - 1) { - if (rtl) - workspaceX1 -= WORKSPACE_CUT_SIZE; - else - workspaceX2 += WORKSPACE_CUT_SIZE; - } - - return x > workspaceX1 && x <= workspaceX2; - }, - - vfunc_get_preferred_height(forWidth) { - if (forWidth < 10) - return [this._porthole.height, this._porthole.height]; - - let themeNode = this.get_theme_node(); - - forWidth = themeNode.adjust_for_width(forWidth); - - let spacing = themeNode.get_length('spacing'); - let nWorkspaces = this._thumbnails.length; - let totalSpacing = (nWorkspaces - 1) * spacing; - - const avail = forWidth - totalSpacing; - - let scale = (avail / nWorkspaces) / this._porthole.width; - - const height = Math.round(this._porthole.height * scale); - - return themeNode.adjust_preferred_height(height, height); - }, - - vfunc_get_preferred_width(forHeight) { - if (forHeight < 10) - return [0, this._porthole.width]; - - let themeNode = this.get_theme_node(); - - let spacing = themeNode.get_length('spacing'); - let nWorkspaces = this._thumbnails.length; - // remove also left/right box padding from the total spacing - let totalSpacing = (nWorkspaces - 3) * spacing; - - const ratio = this._porthole.height / this._porthole.width; - - const tmbWidth = themeNode.adjust_for_height(forHeight) / ratio; - - const naturalWidth = Math.round( - this._thumbnails.reduce((accumulator, thumbnail) => { - const progress = 1 - thumbnail.collapse_fraction; - const width = tmbWidth * progress; - return accumulator + width; - }, 0) - ); - - return themeNode.adjust_preferred_width(totalSpacing, naturalWidth); - }, - - vfunc_allocate(box) { - this.set_allocation(box); - - let rtl = Clutter.get_default_text_direction() === Clutter.TextDirection.RTL; - - if (this._thumbnails.length === 0) // not visible - return; - - let themeNode = this.get_theme_node(); - box = themeNode.get_content_box(box); - - const portholeWidth = this._porthole.width; - const portholeHeight = this._porthole.height; - const spacing = themeNode.get_length('spacing'); - - /* const nWorkspaces = this._thumbnails.length; */ - - // Compute the scale we'll need once everything is updated, - // unless we are currently transitioning - if (this._expandFraction === 1) { - // remove size "breathing" during adding/removing workspaces - - /* const totalSpacing = (nWorkspaces - 1) * spacing; - const availableWidth = (box.get_width() - totalSpacing) / nWorkspaces; - - const hScale = availableWidth / portholeWidth; */ - const hScale = box.get_width() / portholeWidth; - const vScale = box.get_height() / portholeHeight; - const newScale = Math.min(hScale, vScale); - - if (newScale !== this._targetScale) { - if (this._targetScale > 0) { - // We don't ease immediately because we need to observe the - // ordering in queueUpdateStates - if workspaces have been - // removed we need to slide them out as the first thing. - this._targetScale = newScale; - this._pendingScaleUpdate = true; - } else { - this._targetScale = this._scale = newScale; - } - - this._queueUpdateStates(); - } - } - - const ratio = portholeWidth / portholeHeight; - const thumbnailFullHeight = Math.round(portholeHeight * this._scale); - const thumbnailWidth = Math.round(thumbnailFullHeight * ratio); - const thumbnailHeight = thumbnailFullHeight * this._expandFraction; - const roundedVScale = thumbnailHeight / portholeHeight; - - let indicatorValue = this._scrollAdjustment.value; - let indicatorUpperWs = Math.ceil(indicatorValue); - let indicatorLowerWs = Math.floor(indicatorValue); - - let indicatorLowerX1 = 0; - let indicatorLowerX2 = 0; - let indicatorUpperX1 = 0; - let indicatorUpperX2 = 0; - - let indicatorThemeNode = this._indicator.get_theme_node(); - let indicatorTopFullBorder = indicatorThemeNode.get_padding(St.Side.TOP) + indicatorThemeNode.get_border_width(St.Side.TOP); - let indicatorBottomFullBorder = indicatorThemeNode.get_padding(St.Side.BOTTOM) + indicatorThemeNode.get_border_width(St.Side.BOTTOM); - let indicatorLeftFullBorder = indicatorThemeNode.get_padding(St.Side.LEFT) + indicatorThemeNode.get_border_width(St.Side.LEFT); - let indicatorRightFullBorder = indicatorThemeNode.get_padding(St.Side.RIGHT) + indicatorThemeNode.get_border_width(St.Side.RIGHT); - - let x = box.x1; - - if (this._dropPlaceholderPos === -1) { - this._dropPlaceholder.allocate_preferred_size( - ...this._dropPlaceholder.get_position()); - - const laters = global.compositor.get_laters(); - laters.add(Meta.LaterType.BEFORE_REDRAW, () => { - this._dropPlaceholder.hide(); - }); - } - - let childBox = new Clutter.ActorBox(); - - for (let i = 0; i < this._thumbnails.length; i++) { - const thumbnail = this._thumbnails[i]; - if (i > 0) - x += spacing - Math.round(thumbnail.collapse_fraction * spacing); - - const y1 = box.y1; - const y2 = y1 + thumbnailHeight; - - if (i === this._dropPlaceholderPos) { - const [, placeholderWidth] = this._dropPlaceholder.get_preferred_width(-1); - childBox.y1 = y1; - childBox.y2 = y2; - - if (rtl) { - childBox.x2 = box.x2 - Math.round(x); - childBox.x1 = box.x2 - Math.round(x + placeholderWidth); - } else { - childBox.x1 = Math.round(x); - childBox.x2 = Math.round(x + placeholderWidth); - } - - this._dropPlaceholder.allocate(childBox); - - const laters = global.compositor.get_laters(); - laters.add(Meta.LaterType.BEFORE_REDRAW, () => { - this._dropPlaceholder.show(); - }); - x += placeholderWidth + spacing; - } - - // We might end up with thumbnailWidth being something like 99.33 - // pixels. To make this work and not end up with a gap at the end, - // we need some thumbnails to be 99 pixels and some 100 pixels width; - // we compute an actual scale separately for each thumbnail. - const x1 = Math.round(x); - const x2 = Math.round(x + thumbnailWidth); - const roundedHScale = (x2 - x1) / portholeWidth; - - // Allocating a scaled actor is funny - x1/y1 correspond to the origin - // of the actor, but x2/y2 are increased by the *unscaled* size. - if (rtl) { - childBox.x2 = box.x2 - x1; - childBox.x1 = box.x2 - (x1 + thumbnailWidth); - } else { - childBox.x1 = x1; - childBox.x2 = x1 + thumbnailWidth; - } - childBox.y1 = y1; - childBox.y2 = y1 + thumbnailHeight; - - thumbnail.setScale(roundedHScale, roundedVScale); - thumbnail.allocate(childBox); - - if (i === indicatorUpperWs) { - indicatorUpperX1 = childBox.x1; - indicatorUpperX2 = childBox.x2; - } - if (i === indicatorLowerWs) { - indicatorLowerX1 = childBox.x1; - indicatorLowerX2 = childBox.x2; - } - - // We round the collapsing portion so that we don't get thumbnails resizing - // during an animation due to differences in rounded, but leave the uncollapsed - // portion unrounded so that non-animating we end up with the right total - x += thumbnailWidth - Math.round(thumbnailWidth * thumbnail.collapse_fraction); - } - - childBox.y1 = box.y1; - childBox.y2 = box.y1 + thumbnailHeight; - - const indicatorX1 = indicatorLowerX1 + - (indicatorUpperX1 - indicatorLowerX1) * (indicatorValue % 1); - const indicatorX2 = indicatorLowerX2 + - (indicatorUpperX2 - indicatorLowerX2) * (indicatorValue % 1); - - childBox.x1 = indicatorX1 - indicatorLeftFullBorder; - childBox.x2 = indicatorX2 + indicatorRightFullBorder; - childBox.y1 -= indicatorTopFullBorder; - childBox.y2 += indicatorBottomFullBorder; - this._indicator.allocate(childBox); - }, - - _updateShouldShow: ThumbnailsBoxVertical._updateShouldShow, -}; - -const WindowClone = { - after__init() { - // Make it transparent and smaller than usual while dragging - this._draggable._dragActorOpacity = 200; - this._draggable._dragActorMaxSize = 150; - }, -}; diff --git a/extensions/46/vertical-workspaces/lib/workspacesView.js b/extensions/46/vertical-workspaces/lib/workspacesView.js deleted file mode 100644 index 5c0d36b..0000000 --- a/extensions/46/vertical-workspaces/lib/workspacesView.js +++ /dev/null @@ -1,1013 +0,0 @@ -/** - * V-Shell (Vertical Workspaces) - * workspacesView.js - * - * @author GdH <G-dH@github.com> - * @copyright 2022 - 2024 - * @license GPL-3.0 - * - */ - -'use strict'; - -import Clutter from 'gi://Clutter'; -import St from 'gi://St'; -import Meta from 'gi://Meta'; -import GObject from 'gi://GObject'; - -import * as Main from 'resource:///org/gnome/shell/ui/main.js'; -import * as Overview from 'resource:///org/gnome/shell/ui/overview.js'; -import * as OverviewControls from 'resource:///org/gnome/shell/ui/overviewControls.js'; -import * as WorkspacesView from 'resource:///org/gnome/shell/ui/workspacesView.js'; - -import * as Util from 'resource:///org/gnome/shell/misc/util.js'; - -let Me; -let opt; - -const ControlsState = OverviewControls.ControlsState; -const FitMode = WorkspacesView.FitMode; - -export const WorkspacesViewModule = class { - constructor(me) { - Me = me; - opt = Me.opt; - - this._firstActivation = true; - this.moduleEnabled = false; - this._overrides = null; - } - - cleanGlobals() { - Me = null; - opt = null; - } - - update(reset) { - this.moduleEnabled = true; - const conflict = false; - - reset = reset || !this.moduleEnabled || conflict; - - // don't touch the original code if module disabled - if (reset && !this._firstActivation) { - this._disableModule(); - } else if (!reset) { - this._firstActivation = false; - this._activateModule(); - } - if (reset && this._firstActivation) - console.debug(' WorkspacesViewModule - Keeping untouched'); - } - - _activateModule() { - if (!this._overrides) - this._overrides = new Me.Util.Overrides(); - - const desktopCubeEnabled = Me.Util.getEnabledExtensions('desktop-cube@schneegans.github.com').length; - const desktopCubeConflict = desktopCubeEnabled && !opt.ORIENTATION && !opt.OVERVIEW_MODE; - - if (!desktopCubeConflict) - this._overrides.addOverride('WorkspacesView', WorkspacesView.WorkspacesView.prototype, WorkspacesViewCommon); - else - this._overrides.removeOverride('WorkspacesView'); - - this._overrides.addOverride('WorkspacesDisplay', WorkspacesView.WorkspacesDisplay.prototype, WorkspacesDisplayCommon); - this._overrides.addOverride('ExtraWorkspaceView', WorkspacesView.ExtraWorkspaceView.prototype, ExtraWorkspaceViewCommon); - this._overrides.addOverride('SecondaryMonitorDisplayCommon', WorkspacesView.SecondaryMonitorDisplay.prototype, SecondaryMonitorDisplayCommon); - - if (opt.ORIENTATION) { - // switch internal workspace orientation in GS - global.workspace_manager.override_workspace_layout(Meta.DisplayCorner.TOPLEFT, false, -1, 1); - this._overrides.addOverride('SecondaryMonitorDisplay', WorkspacesView.SecondaryMonitorDisplay.prototype, SecondaryMonitorDisplayVertical); - } else { - global.workspace_manager.override_workspace_layout(Meta.DisplayCorner.TOPLEFT, false, 1, -1); - this._overrides.addOverride('SecondaryMonitorDisplay', WorkspacesView.SecondaryMonitorDisplay.prototype, SecondaryMonitorDisplayHorizontal); - } - - console.debug(' WorkspacesViewModule - Activated'); - } - - _disableModule() { - global.workspace_manager.override_workspace_layout(Meta.DisplayCorner.TOPLEFT, false, 1, -1); - if (this._overrides) - this._overrides.removeAll(); - this._overrides = null; - - console.debug(' WorkspacesViewModule - Disabled'); - } -}; - -const WorkspacesViewCommon = { - _getFirstFitSingleWorkspaceBox(box, spacing, vertical) { - let [width, height] = box.get_size(); - const [workspace] = this._workspaces; - - const rtl = this.text_direction === Clutter.TextDirection.RTL; - const adj = this._scrollAdjustment; - const currentWorkspace = vertical || !rtl - ? adj.value : adj.upper - adj.value - 1; - - // Single fit mode implies centered too - let [x1, y1] = box.get_origin(); - const [, workspaceWidth] = workspace ? workspace.get_preferred_width(Math.floor(height)) : [0, width]; - const [, workspaceHeight] = workspace ? workspace.get_preferred_height(workspaceWidth) : [0, height]; - - if (vertical) { - x1 += (width - workspaceWidth) / 2; - y1 -= currentWorkspace * (workspaceHeight + spacing); - } else { - x1 += (width - workspaceWidth) / 2; - x1 -= currentWorkspace * (workspaceWidth + spacing); - } - - const fitSingleBox = new Clutter.ActorBox({ x1, y1 }); - fitSingleBox.set_size(workspaceWidth, workspaceHeight); - - return fitSingleBox; - }, - - // set spacing between ws previews - _getSpacing(box, fitMode, vertical) { - const [width, height] = box.get_size(); - const [workspace] = this._workspaces; - - if (!workspace) - return 0; - - let availableSpace; - let workspaceSize; - if (vertical) { - [, workspaceSize] = workspace.get_preferred_height(width); - availableSpace = height; - } else { - [, workspaceSize] = workspace.get_preferred_width(height); - availableSpace = width; - } - - const spacing = (availableSpace - workspaceSize * 0.4) * (1 - fitMode); - const { scaleFactor } = St.ThemeContext.get_for_stage(global.stage); - return Math.clamp(spacing, - opt.WORKSPACE_MIN_SPACING * scaleFactor, - opt.WORKSPACE_MAX_SPACING * scaleFactor); - }, - - // this function has duplicate in OverviewControls so we use one function for both to avoid issues with syncing them - _getFitModeForState(state) { - return _getFitModeForState(state); - }, - - // normal view 0, spread windows 1 - _getWorkspaceModeForOverviewState(state) { - - switch (state) { - case ControlsState.HIDDEN: - return 0; - case ControlsState.WINDOW_PICKER: - return opt.WORKSPACE_MODE; - case ControlsState.APP_GRID: - return (this._monitorIndex !== global.display.get_primary_monitor() || !opt.WS_ANIMATION) && !opt.OVERVIEW_MODE ? 1 : 0; - } - - return 0; - }, - - _updateVisibility() { - // visibility handles _updateWorkspacesState() - }, - - // disable scaling and hide inactive workspaces - _updateWorkspacesState() { - const adj = this._scrollAdjustment; - const fitMode = this._fitModeAdjustment.value; - - let { initialState, finalState, progress, currentState } = - this._overviewAdjustment.getStateTransitionParams(); - - const workspaceMode = (1 - fitMode) * Util.lerp( - this._getWorkspaceModeForOverviewState(initialState), - this._getWorkspaceModeForOverviewState(finalState), - progress); - - const primaryMonitor = Main.layoutManager.primaryMonitor.index; - - const wsScrollProgress = adj.value % 1; - const secondaryMonitor = this._monitorIndex !== global.display.get_primary_monitor(); - const blockSecondaryAppGrid = opt.OVERVIEW_MODE && currentState > 1; - - // Hide inactive workspaces - this._workspaces.forEach((w, index) => { - if (!(blockSecondaryAppGrid && secondaryMonitor)) - w.stateAdjustment.value = workspaceMode; - - let distance = adj.value - index; - const distanceToCurrentWorkspace = Math.abs(distance); - - const scaleProgress = 1 - Math.clamp(distanceToCurrentWorkspace, 0, 1); - // const scale = Util.lerp(0.94, 1, scaleProgress); - // w.set_scale(scale, scale); - - // if we disable workspaces that we can't or don't need to see, transition animations will be noticeably smoother - // only the current ws needs to be visible during overview transition animations - // and only current and adjacent ws when switching ws - w.visible = - (this._animating && wsScrollProgress && distanceToCurrentWorkspace <= (opt.NUMBER_OF_VISIBLE_NEIGHBORS + 1)) || - scaleProgress === 1 || - (opt.WORKSPACE_MAX_SPACING >= opt.WS_MAX_SPACING_OFF_SCREEN && - distanceToCurrentWorkspace <= opt.NUMBER_OF_VISIBLE_NEIGHBORS && - currentState === ControlsState.WINDOW_PICKER - ) || - (this._monitorIndex !== primaryMonitor && distanceToCurrentWorkspace <= opt.NUMBER_OF_VISIBLE_NEIGHBORS) || - (!opt.WS_ANIMATION && distanceToCurrentWorkspace < opt.NUMBER_OF_VISIBLE_NEIGHBORS) || - (distanceToCurrentWorkspace <= opt.NUMBER_OF_VISIBLE_NEIGHBORS && - currentState <= ControlsState.WINDOW_PICKER && - (initialState < ControlsState.APP_GRID && finalState < ControlsState.APP_GRID) - ); - - // after transition from APP_GRID to WINDOW_PICKER state, - // adjacent workspaces are hidden and we need them to show up - // make them visible during animation can impact smoothness of the animation - // so we show them after the animation finished, move them to their position from outside of the monitor - if (currentState === ControlsState.WINDOW_PICKER && !w.visible && distanceToCurrentWorkspace <= opt.NUMBER_OF_VISIBLE_NEIGHBORS && initialState === ControlsState.APP_GRID) { - w.remove_all_transitions(); - w.visible = true; - const directionNext = distance > 0; - if (!opt.ORIENTATION) { - const width = w.width * 0.6 * opt.WS_PREVIEW_SCALE; - w.translation_x = directionNext ? -width : width; - } - if (opt.ORIENTATION) { - const height = w.height * 0.6 * opt.WS_PREVIEW_SCALE; - w.translation_y = directionNext ? -height : height; - } - - w.opacity = 10; - w.get_parent().set_child_below_sibling(w, null); - w.ease({ - duration: 300, - translation_x: 0, - translation_y: 0, - opacity: 255, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - - // force ws preview bg corner radiuses where GS doesn't do it - if (opt.SHOW_WS_PREVIEW_BG && opt.OVERVIEW_MODE === 1 && distanceToCurrentWorkspace < 2) - w._background._updateBorderRadius(Math.min(1, w._overviewAdjustment.value)); - - - // hide workspace background - if (!opt.SHOW_WS_PREVIEW_BG && w._background.opacity) - w._background.opacity = 0; - }); - }, - - exposeWindows(workspaceIndex = null, callback) { - let adjustments = []; - if (workspaceIndex === null) { - this._workspaces.forEach(ws => { - adjustments.push(ws._background._stateAdjustment); - }); - } else { - adjustments.push(this._workspaces[workspaceIndex]._background._stateAdjustment); - } - - opt.WORKSPACE_MODE = 1; - adjustments.forEach(adj => { - if (adj.value === 0) { - adj.value = 0; - adj.ease(1, { - duration: 200, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - if (callback) - callback(); - }, - }); - } - }); - }, -}; - -const SecondaryMonitorDisplayCommon = { - exposeWindows(...args) { - this._workspacesView.exposeWindows(...args); - }, -}; - -const SecondaryMonitorDisplayVertical = { - _getThumbnailParamsForState(state) { - - let opacity, scale, translationX; - switch (state) { - case ControlsState.HIDDEN: - opacity = 255; - scale = 1; - translationX = 0; - if (!Main.layoutManager._startingUp && (!opt.SHOW_WS_PREVIEW_BG || opt.OVERVIEW_MODE2)) - translationX = this._thumbnails.width * (opt.SEC_WS_TMB_LEFT ? -1 : 1); - - break; - case ControlsState.WINDOW_PICKER: - case ControlsState.APP_GRID: - opacity = 255; - scale = 1; - translationX = 0; - break; - default: - opacity = 255; - scale = 1; - translationX = 0; - break; - } - - return { opacity, scale, translationX }; - }, - - _getWorkspacesBoxForState(state, box, workArea, wsTmbWidth, spacing) { - let workspaceBox = box.copy(); - - if ( - (state === ControlsState.WINDOW_PICKER || state === ControlsState.APP_GRID) && - !(opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE) - ) { - workspaceBox = workArea.copy(); - const [startX, startY] = workspaceBox.get_origin(); - let [width, height] = workspaceBox.get_size(); - - let wsBoxWidth = width - (wsTmbWidth ? wsTmbWidth + spacing : 0) - 2 * spacing; - let wsBoxHeight = height - 2 * spacing; - - const ratio = width / height; - let wRatio = wsBoxWidth / wsBoxHeight; - let scale = ratio / wRatio; - - if (scale > 1) { - wsBoxHeight /= scale; - wsBoxWidth = wsBoxHeight * ratio; - } else { - wsBoxWidth *= scale; - wsBoxHeight = wsBoxWidth / ratio; - } - - // height decides the actual size, ratio is given by the workArea - wsBoxHeight = Math.round(wsBoxHeight * opt.SEC_WS_PREVIEW_SCALE); - wsBoxWidth = Math.round(wsBoxWidth * opt.SEC_WS_PREVIEW_SCALE); - - let offset = Math.round(width - wsTmbWidth - wsBoxWidth - spacing) / 2; - - const wsbX = startX + opt.SEC_WS_TMB_LEFT - ? wsTmbWidth + spacing + offset - : offset; - - const wsbY = Math.round((startY + height - wsBoxHeight) / 2); - - workspaceBox.set_origin(wsbX, wsbY); - workspaceBox.set_size(wsBoxWidth, wsBoxHeight); - } - - return workspaceBox; - }, - - _getWorkAreaBox(box) { - if (!opt.SEC_WS_PREVIEW_SHIFT || !Main.panel.visible) - return box; - - const workArea = box.copy(); - const panelHeight = Main.panel.height; - workArea.y1 += opt.PANEL_POSITION_TOP ? panelHeight : 0; - workArea.y2 -= opt.PANEL_POSITION_BOTTOM ? panelHeight : 0; - - return workArea; - }, - - vfunc_allocate(box) { - this.set_allocation(box); - - const themeNode = this.get_theme_node(); - const contentBox = themeNode.get_content_box(box); - - const workArea = this._getWorkAreaBox(contentBox); - - let [width, height] = workArea.get_size(); - let [startX, startY] = workArea.get_origin(); - - const spacing = opt.SPACING; - - let wsTmbWidth = 0; - let wsTmbHeight = 0; - this._thumbnails.visible = !opt.SEC_WS_TMB_HIDDEN; - if (this._thumbnails.visible) { - wsTmbWidth = Math.round(width * opt.SEC_MAX_THUMBNAIL_SCALE); - - let totalTmbSpacing; - [totalTmbSpacing, wsTmbHeight] = this._thumbnails.get_preferred_height(wsTmbWidth); - wsTmbHeight += totalTmbSpacing; - - const thumbnailsHeightMax = height - spacing; - - if (wsTmbHeight > thumbnailsHeightMax) { - wsTmbHeight = thumbnailsHeightMax; - wsTmbWidth = Math.round(this._thumbnails.get_preferred_width(wsTmbHeight)[1]); - } - - let wsTmbX = opt.SEC_WS_TMB_LEFT - ? startX + spacing - : startX + width - wsTmbWidth - spacing; - - let offset = (height - wsTmbHeight) / 2; - const wsTmbY = startY + Math.round(offset - opt.SEC_WS_TMB_POSITION_ADJUSTMENT * (offset - spacing)); - - const childBox = new Clutter.ActorBox(); - childBox.set_origin(wsTmbX, wsTmbY); - childBox.set_size(wsTmbWidth, wsTmbHeight); - this._thumbnails.allocate(childBox); - } - - const { - currentState, initialState, finalState, transitioning, progress, - } = this._overviewAdjustment.getStateTransitionParams(); - - let workspacesBox; - const workspaceParams = [contentBox, workArea, wsTmbWidth, spacing]; - if (!transitioning) { - workspacesBox = - this._getWorkspacesBoxForState(currentState, ...workspaceParams); - } else { - const initialBox = - this._getWorkspacesBoxForState(initialState, ...workspaceParams); - const finalBox = - this._getWorkspacesBoxForState(finalState, ...workspaceParams); - workspacesBox = initialBox.interpolate(finalBox, progress); - } - this._workspacesView.allocate(workspacesBox); - }, - - _updateThumbnailVisibility() { - if (opt.OVERVIEW_MODE2) - this.set_child_above_sibling(this._thumbnails, null); - - const visible = !opt.SEC_WS_TMB_HIDDEN; - - if (this._thumbnails.visible === visible) - return; - - this._thumbnails.show(); - this._updateThumbnailParams(); - this._thumbnails.ease_property('expand-fraction', visible ? 1 : 0, { - duration: Overview.ANIMATION_TIME, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - onComplete: () => { - this._thumbnails.visible = visible; - this._thumbnails._indicator.visible = visible; - }, - }); - }, - - _updateThumbnailParams() { - if (opt.SEC_WS_TMB_HIDDEN) - return; - - // workaround for upstream bug - secondary thumbnails boxes don't catch 'showing' signal on the shell startup and don't populate the box with thumbnails - // the tmbBox contents is also destroyed when overview state adjustment gets above 1 when swiping gesture from window picker to app grid - if (!this._thumbnails._thumbnails.length) - this._thumbnails._createThumbnails(); - - const { initialState, finalState, progress } = - this._overviewAdjustment.getStateTransitionParams(); - - const initialParams = this._getThumbnailParamsForState(initialState); - const finalParams = this._getThumbnailParamsForState(finalState); - - /* const opacity = - Util.lerp(initialParams.opacity, finalParams.opacity, progress); - const scale = - Util.lerp(initialParams.scale, finalParams.scale, progress);*/ - - // OVERVIEW_MODE 2 should animate dash and wsTmbBox only if WORKSPACE_MODE === 0 (windows not spread) - const animateOverviewMode2 = opt.OVERVIEW_MODE2 && !(finalState === 1 && opt.WORKSPACE_MODE); - const translationX = !Main.layoutManager._startingUp && ((!opt.SHOW_WS_PREVIEW_BG && !opt.OVERVIEW_MODE2) || animateOverviewMode2) - ? Util.lerp(initialParams.translationX, finalParams.translationX, progress) - : 0; - - this._thumbnails.set({ - opacity: 255, - // scale_x: scale, - // scale_y: scale, - translation_x: translationX, - }); - }, - - _updateWorkspacesView() { - if (this._workspacesView) - this._workspacesView.destroy(); - - if (this._settings.get_boolean('workspaces-only-on-primary')) { - opt.SEC_WS_TMB_HIDDEN = true; - this._workspacesView = new WorkspacesView.ExtraWorkspaceView( - this._monitorIndex, - this._overviewAdjustment); - } else { - opt.SEC_WS_TMB_HIDDEN = !opt.SHOW_SEC_WS_TMB; - this._workspacesView = new WorkspacesView.WorkspacesView( - this._monitorIndex, - this._controls, - this._scrollAdjustment, - // Secondary monitors don't need FitMode.ALL since there is workspace switcher always visible - // this._fitModeAdjustment, - new St.Adjustment({ - actor: this, - value: 0, // FitMode.SINGLE, - lower: 0, // FitMode.SINGLE, - upper: 0, // FitMode.SINGLE, - }), - // secondaryOverviewAdjustment); - this._overviewAdjustment); - } - this.add_child(this._workspacesView); - this._thumbnails.opacity = 0; - }, -}; - -const SecondaryMonitorDisplayHorizontal = { - _getThumbnailParamsForState(state) { - // const { ControlsState } = OverviewControls; - - let opacity, scale, translationY; - switch (state) { - case ControlsState.HIDDEN: - opacity = 255; - scale = 1; - translationY = 0; - if (!Main.layoutManager._startingUp && (!opt.SHOW_WS_PREVIEW_BG || opt.OVERVIEW_MODE2)) - translationY = this._thumbnails.height * (opt.SEC_WS_TMB_TOP ? -1 : 1); - - break; - case ControlsState.WINDOW_PICKER: - case ControlsState.APP_GRID: - opacity = 255; - scale = 1; - translationY = 0; - break; - default: - opacity = 255; - scale = 1; - translationY = 0; - break; - } - - return { opacity, scale, translationY }; - }, - - _getWorkspacesBoxForState(state, box, workArea, wsTmbHeight, spacing) { - let workspaceBox = box.copy(); - - if ( - (state === ControlsState.WINDOW_PICKER || state === ControlsState.APP_GRID) && - !(opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE) - ) { - workspaceBox = workArea.copy(); - const [startX, startY] = workspaceBox.get_origin(); - let [width, height] = workspaceBox.get_size(); - - let wsBoxWidth = width - 2 * spacing; - let wsBoxHeight = height - (wsTmbHeight ? wsTmbHeight + spacing : 0) - 2 * spacing; - - const ratio = width / height; - let wRatio = wsBoxWidth / wsBoxHeight; - let scale = ratio / wRatio; - - if (scale > 1) { - wsBoxHeight /= scale; - wsBoxWidth = wsBoxHeight * ratio; - } else { - wsBoxWidth *= scale; - wsBoxHeight = wsBoxWidth / ratio; - } - - // height decides the actual size, ratio is given by the workArea - wsBoxHeight = Math.round(wsBoxHeight * opt.SEC_WS_PREVIEW_SCALE); - wsBoxWidth = Math.round(wsBoxWidth * opt.SEC_WS_PREVIEW_SCALE); - - let offset = Math.round(height - wsTmbHeight - wsBoxHeight - spacing) / 2; - - const wsbX = Math.round((startX + width - wsBoxWidth) / 2); - - const wsbY = startY + opt.SEC_WS_TMB_TOP - ? wsTmbHeight + spacing + offset - : offset; - - workspaceBox.set_origin(wsbX, wsbY); - workspaceBox.set_size(wsBoxWidth, wsBoxHeight); - } - - return workspaceBox; - }, - - _getWorkAreaBox: SecondaryMonitorDisplayVertical._getWorkAreaBox, - - vfunc_allocate(box) { - this.set_allocation(box); - - const themeNode = this.get_theme_node(); - const contentBox = themeNode.get_content_box(box); - - const workArea = this._getWorkAreaBox(contentBox); - - let [width, height] = workArea.get_size(); - let [startX, startY] = workArea.get_origin(); - - const spacing = opt.SPACING; - - let wsTmbWidth = 0; - let wsTmbHeight = 0; - this._thumbnails.visible = !opt.SEC_WS_TMB_HIDDEN; - if (this._thumbnails.visible) { - wsTmbHeight = Math.round(height * opt.SEC_MAX_THUMBNAIL_SCALE); - - let totalTmbSpacing; - [totalTmbSpacing, wsTmbWidth] = this._thumbnails.get_preferred_width(wsTmbHeight); - wsTmbWidth += totalTmbSpacing; - - const thumbnailsWidthMax = width - 2 * spacing; - - if (wsTmbWidth > thumbnailsWidthMax) { - wsTmbWidth = thumbnailsWidthMax; - wsTmbHeight = Math.round(this._thumbnails.get_preferred_height(wsTmbWidth)[1]); - } - - let wsTmbY = opt.SEC_WS_TMB_TOP - ? startY + spacing - : startY + height - wsTmbHeight - spacing; - - let offset = (width - wsTmbWidth) / 2; - const wsTmbX = startX + Math.round(offset - opt.SEC_WS_TMB_POSITION_ADJUSTMENT * (offset - spacing)); - - const childBox = new Clutter.ActorBox(); - childBox.set_origin(wsTmbX, wsTmbY); - childBox.set_size(wsTmbWidth, wsTmbHeight); - this._thumbnails.allocate(childBox); - } - - const { - currentState, initialState, finalState, transitioning, progress, - } = this._overviewAdjustment.getStateTransitionParams(); - - let workspacesBox; - const workspaceParams = [contentBox, workArea, wsTmbHeight, spacing]; - if (!transitioning) { - workspacesBox = - this._getWorkspacesBoxForState(currentState, ...workspaceParams); - } else { - const initialBox = - this._getWorkspacesBoxForState(initialState, ...workspaceParams); - const finalBox = - this._getWorkspacesBoxForState(finalState, ...workspaceParams); - workspacesBox = initialBox.interpolate(finalBox, progress); - } - this._workspacesView.allocate(workspacesBox); - }, - - _updateThumbnailVisibility: SecondaryMonitorDisplayVertical._updateThumbnailVisibility, - - _updateThumbnailParams() { - if (opt.SEC_WS_TMB_HIDDEN) - return; - - // workaround for upstream bug - secondary thumbnails boxes don't catch 'showing' signal on the shell startup and don't populate the box with thumbnails - // the tmbBox contents is also destroyed when overview state adjustment gets above 1 when swiping gesture from window picker to app grid - if (!this._thumbnails._thumbnails.length) - this._thumbnails._createThumbnails(); - - const { initialState, finalState, progress } = - this._overviewAdjustment.getStateTransitionParams(); - - const initialParams = this._getThumbnailParamsForState(initialState); - const finalParams = this._getThumbnailParamsForState(finalState); - - /* const opacity = - Util.lerp(initialParams.opacity, finalParams.opacity, progress); - const scale = - Util.lerp(initialParams.scale, finalParams.scale, progress);*/ - - // OVERVIEW_MODE 2 should animate dash and wsTmbBox only if WORKSPACE_MODE === 0 (windows not spread) - const animateOverviewMode2 = opt.OVERVIEW_MODE2 && !(finalState === 1 && opt.WORKSPACE_MODE); - const translationY = !Main.layoutManager._startingUp && ((!opt.SHOW_WS_PREVIEW_BG && !opt.OVERVIEW_MODE2) || animateOverviewMode2) - ? Util.lerp(initialParams.translationY, finalParams.translationY, progress) - : 0; - - this._thumbnails.set({ - opacity: 255, - // scale_x: scale, - // scale_y: scale, - translation_y: translationY, - }); - }, - - _updateWorkspacesView() { - if (this._workspacesView) - this._workspacesView.destroy(); - - if (this._settings.get_boolean('workspaces-only-on-primary')) { - opt.SEC_WS_TMB_HIDDEN = true; - this._workspacesView = new WorkspacesView.ExtraWorkspaceView( - this._monitorIndex, - this._overviewAdjustment); - } else { - opt.SEC_WS_TMB_HIDDEN = !opt.SHOW_SEC_WS_TMB; - this._workspacesView = new WorkspacesView.WorkspacesView( - this._monitorIndex, - this._controls, - this._scrollAdjustment, - // Secondary monitors don't need FitMode.ALL since there is workspace switcher always visible - // this._fitModeAdjustment, - new St.Adjustment({ - actor: this, - value: 0, // FitMode.SINGLE, - lower: 0, // FitMode.SINGLE, - upper: 0, // FitMode.SINGLE, - }), - // secondaryOverviewAdjustment); - this._overviewAdjustment); - } - this.add_child(this._workspacesView); - this._thumbnails.opacity = 0; - }, -}; - -const ExtraWorkspaceViewCommon = { - _updateWorkspaceMode() { - const overviewState = this._overviewAdjustment.value; - - const progress = Math.clamp(overviewState, - ControlsState.HIDDEN, - opt.OVERVIEW_MODE && !opt.WORKSPACE_MODE ? ControlsState.HIDDEN : ControlsState.WINDOW_PICKER); - - this._workspace.stateAdjustment.value = progress; - - // force ws preview bg corner radiuses where GS doesn't do it - if (opt.SHOW_WS_PREVIEW_BG && opt.OVERVIEW_MODE === 1) - this._workspace._background._updateBorderRadius(Math.min(1, this._workspace._overviewAdjustment.value)); - - - // hide workspace background - if (!opt.SHOW_WS_PREVIEW_BG && this._workspace._background.opacity) - this._workspace._background.opacity = 0; - }, - - exposeWindows() { - const adjustment = this._workspace._background._stateAdjustment; - opt.WORKSPACE_MODE = 1; - if (adjustment.value === 0) { - adjustment.value = 0; - adjustment.ease(1, { - duration: 200, - mode: Clutter.AnimationMode.EASE_OUT_QUAD, - }); - } - }, -}; - -const WorkspacesDisplayCommon = { - _updateWorkspacesViews() { - for (let i = 0; i < this._workspacesViews.length; i++) - this._workspacesViews[i].destroy(); - - this._primaryIndex = Main.layoutManager.primaryIndex; - this._workspacesViews = []; - let monitors = Main.layoutManager.monitors; - for (let i = 0; i < monitors.length; i++) { - let view; - if (i === this._primaryIndex) { - view = new WorkspacesView.WorkspacesView(i, - this._controls, - this._scrollAdjustment, - this._fitModeAdjustment, - this._overviewAdjustment); - - view.visible = this._primaryVisible; - this.bind_property('opacity', view, 'opacity', GObject.BindingFlags.SYNC_CREATE); - this.add_child(view); - } else { - view = new WorkspacesView.SecondaryMonitorDisplay(i, - this._controls, - this._scrollAdjustment, - // Secondary monitors don't need FitMode.ALL since there is workspace switcher always visible - // this._fitModeAdjustment, - new St.Adjustment({ - actor: this, - value: 0, // FitMode.SINGLE, - lower: 0, // FitMode.SINGLE, - upper: 0, // FitMode.SINGLE, - }), - this._overviewAdjustment); - Main.layoutManager.overviewGroup.add_child(view); - - if (opt.CLICK_EMPTY_CLOSE) { - // Allow users to close the overview by clicking on an empty space on the secondary monitor - // The primary monitor overview is handled in the overviewControls - const clickAction = new Clutter.ClickAction(); - clickAction.connect('clicked', () => { - Main.overview.hide(); - }); - view.reactive = true; - view.add_action(clickAction); - } - } - - this._workspacesViews.push(view); - } - }, - - _onScrollEvent(actor, event) { - if (this._swipeTracker.canHandleScrollEvent(event)) - return Clutter.EVENT_PROPAGATE; - - if (!this.mapped) - return Clutter.EVENT_PROPAGATE; - - if (this._workspacesOnlyOnPrimary && - this._getMonitorIndexForEvent(event) !== this._primaryIndex) - return Clutter.EVENT_PROPAGATE; - - if (opt.PANEL_MODE === 1) { - const panelBox = Main.layoutManager.panelBox; - const [, y] = global.get_pointer(); - if (y > panelBox.allocation.y1 && y < panelBox.allocation.y2) - return Clutter.EVENT_STOP; - } - - if (Me.Util.isShiftPressed()) { - let direction = Me.Util.getScrollDirection(event); - if (direction === null || (Date.now() - this._lastScrollTime) < 150) - return Clutter.EVENT_STOP; - this._lastScrollTime = Date.now(); - - if (direction === Clutter.ScrollDirection.UP) - direction = -1; - - else if (direction === Clutter.ScrollDirection.DOWN) - direction = 1; - else - direction = 0; - - if (direction) { - Me.Util.reorderWorkspace(direction); - // make all workspaces on primary monitor visible for case the new position is hidden - const primaryMonitorIndex = global.display.get_primary_monitor(); - Main.overview._overview._controls._workspacesDisplay._workspacesViews[primaryMonitorIndex]._workspaces.forEach(w => { - w.visible = true; - }); - return Clutter.EVENT_STOP; - } - } - - return Main.wm.handleWorkspaceScroll(event); - }, - - _onKeyPressEvent(actor, event) { - const symbol = event.get_key_symbol(); - /* const { ControlsState } = OverviewControls; - if (this._overviewAdjustment.value !== ControlsState.WINDOW_PICKER && symbol !== Clutter.KEY_space) - return Clutter.EVENT_PROPAGATE;*/ - - /* if (!this.reactive) - return Clutter.EVENT_PROPAGATE; */ - const { workspaceManager } = global; - const vertical = workspaceManager.layout_rows === -1; - const rtl = this.get_text_direction() === Clutter.TextDirection.RTL; - const state = this._overviewAdjustment.value; - - let which; - switch (symbol) { - case Clutter.KEY_Return: - case Clutter.KEY_KP_Enter: - if (Me.Util.isCtrlPressed()) { - Main.ctrlAltTabManager._items.forEach(i => { - if (i.sortGroup === 1 && i.name === 'Dash') - Main.ctrlAltTabManager.focusGroup(i); - }); - } - return Clutter.EVENT_STOP; - case Clutter.KEY_Page_Up: - if (vertical) - which = Meta.MotionDirection.UP; - else if (rtl) - which = Meta.MotionDirection.RIGHT; - else - which = Meta.MotionDirection.LEFT; - break; - case Clutter.KEY_Page_Down: - if (vertical) - which = Meta.MotionDirection.DOWN; - else if (rtl) - which = Meta.MotionDirection.LEFT; - else - which = Meta.MotionDirection.RIGHT; - break; - case Clutter.KEY_Home: - which = 0; - break; - case Clutter.KEY_End: - which = workspaceManager.n_workspaces - 1; - break; - case Clutter.KEY_space: - if (Me.Util.isCtrlPressed() && Me.Util.isShiftPressed()) { - Me.Util.openPreferences(); - } else if (Me.Util.isAltPressed()) { - Main.ctrlAltTabManager._items.forEach(i => { - if (i.sortGroup === 1 && i.name === 'Dash') - Main.ctrlAltTabManager.focusGroup(i); - }); - } else if (Me.Util.getEnabledExtensions('extensions-search-provider').length && Me.Util.isCtrlPressed()) { - Me.Util.activateSearchProvider(Me.ESP_PREFIX); - } else if (Me.Util.getEnabledExtensions('windows-search-provider').length) { - Me.Util.activateSearchProvider(Me.WSP_PREFIX); - } - - return Clutter.EVENT_STOP; - case Clutter.KEY_Down: - case Clutter.KEY_Left: - case Clutter.KEY_Right: - case Clutter.KEY_Up: - case Clutter.KEY_Tab: - if (Main.overview.searchController.searchActive) { - Main.overview.searchEntry.grab_key_focus(); - } else if (opt.OVERVIEW_MODE2 && !opt.WORKSPACE_MODE && state === 1) { - // expose windows by "clicking" on ws thumbnail - // in this case overview stateAdjustment will be used for transition - Main.overview._overview.controls._thumbnailsBox._activateThumbnailAtPoint(0, 0, global.get_current_time(), true); - Main.ctrlAltTabManager._items.forEach(i => { - if (i.sortGroup === 1 && i.name === 'Windows') - Main.ctrlAltTabManager.focusGroup(i); - }); - } else if (opt.OVERVIEW_MODE && !opt.WORKSPACE_MODE && state === 1) { - // expose windows for OVERVIEW_MODE 1 - const wsIndex = global.workspace_manager.get_active_workspace().index(); - // after expose animation activate keyboard for window selection - const callback = Me.Util.activateKeyboardForWorkspaceView; - this._workspacesViews.forEach( - view => { - view.exposeWindows(wsIndex, callback); - } - ); - } else { - if (state === 2) - return Clutter.EVENT_PROPAGATE; - Me.Util.activateKeyboardForWorkspaceView(); - } - - return Clutter.EVENT_STOP; - default: - return Clutter.EVENT_PROPAGATE; - } - - if (state === 2) - return Clutter.EVENT_PROPAGATE; - - let ws; - if (which < 0) - // Negative workspace numbers are directions - ws = workspaceManager.get_active_workspace().get_neighbor(which); - else - // Otherwise it is a workspace index - ws = workspaceManager.get_workspace_by_index(which); - - if (Me.Util.isShiftPressed()) { - let direction; - if (which === Meta.MotionDirection.UP || which === Meta.MotionDirection.LEFT) - direction = -1; - else if (which === Meta.MotionDirection.DOWN || which === Meta.MotionDirection.RIGHT) - direction = 1; - if (direction) - Me.Util.reorderWorkspace(direction); - // make all workspaces on primary monitor visible for case the new position is hidden - Main.overview._overview._controls._workspacesDisplay._workspacesViews[0]._workspaces.forEach(w => { - w.visible = true; - }); - return Clutter.EVENT_STOP; - } - - if (ws) - Main.wm.actionMoveWorkspace(ws); - - return Clutter.EVENT_STOP; - }, -}; - -// same copy of this function should be available in OverviewControls and WorkspacesView -function _getFitModeForState(state) { - switch (state) { - case ControlsState.HIDDEN: - case ControlsState.WINDOW_PICKER: - return FitMode.SINGLE; - case ControlsState.APP_GRID: - if (opt.WS_ANIMATION && opt.SHOW_WS_TMB) - return FitMode.ALL; - else - return FitMode.SINGLE; - default: - return FitMode.SINGLE; - } -} |