From 42476f340fe167c140e4708cd12066f27c1dc0c0 Mon Sep 17 00:00:00 2001 From: Odei Maiz <33152403+odeimaiz@users.noreply.github.com> Date: Fri, 29 Nov 2024 14:15:47 +0100 Subject: [PATCH] =?UTF-8?q?=E2=9C=A8=20[Frontend]=20Trash=20bin=20(#6590)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../source/class/osparc/dashboard/CardBase.js | 14 +- .../osparc/dashboard/ContextBreadcrumbs.js | 13 +- .../osparc/dashboard/FolderButtonItem.js | 36 +- .../class/osparc/dashboard/GridButtonItem.js | 3 +- .../class/osparc/dashboard/ListButtonItem.js | 3 +- .../osparc/dashboard/ResourceBrowserBase.js | 22 +- .../dashboard/ResourceContainerManager.js | 46 +- .../class/osparc/dashboard/ResourceFilter.js | 83 ++++ .../class/osparc/dashboard/ServiceBrowser.js | 6 +- .../class/osparc/dashboard/StudyBrowser.js | 460 +++++++++++++++--- .../osparc/dashboard/StudyBrowserHeader.js | 204 ++++---- .../class/osparc/dashboard/TemplateBrowser.js | 6 +- .../osparc/dashboard/WorkspaceButtonItem.js | 81 ++- .../dashboard/WorkspacesAndFoldersTree.js | 2 +- .../dashboard/WorkspacesAndFoldersTreeItem.js | 1 + .../source/class/osparc/data/Resources.js | 74 ++- .../source/class/osparc/data/model/Folder.js | 2 +- .../class/osparc/data/model/IframeHandler.js | 2 +- .../class/osparc/data/model/Workspace.js | 16 +- .../source/class/osparc/store/Folders.js | 58 +++ .../source/class/osparc/store/StaticInfo.js | 10 + .../client/source/class/osparc/store/Store.js | 39 +- .../source/class/osparc/store/Workspaces.js | 118 ++++- 23 files changed, 1062 insertions(+), 237 deletions(-) diff --git a/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js b/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js index acb3fe35386..3bcd200c2ee 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js @@ -395,6 +395,14 @@ qx.Class.define("osparc.dashboard.CardBase", { return this.getResourceType() === resourceType; }, + isItemNotClickable: function() { + const studyBrowserContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + return ( + this.getBlocked() === true || // It could be blocked by IN_USE or UNKNOWN_SERVICE + (this.isResourceType("study") && (studyBrowserContext === "trash")) // It could a trashed study + ); + }, + __applyResourceData: function(resourceData) { let uuid = null; let owner = null; @@ -776,9 +784,9 @@ qx.Class.define("osparc.dashboard.CardBase", { if (moveToButton) { moveToButton.setEnabled(osparc.study.Utils.canMoveTo(resourceData)); } - const deleteButton = menuButtons.find(menuBtn => "deleteButton" in menuBtn); - if (deleteButton) { - deleteButton.setEnabled(osparc.study.Utils.canBeDeleted(resourceData)); + const trashButton = menuButtons.find(menuBtn => "trashButton" in menuBtn); + if (trashButton) { + trashButton.setEnabled(osparc.study.Utils.canBeDeleted(resourceData)); } } }, diff --git a/services/static-webserver/client/source/class/osparc/dashboard/ContextBreadcrumbs.js b/services/static-webserver/client/source/class/osparc/dashboard/ContextBreadcrumbs.js index dba858ffad8..4358bc68848 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/ContextBreadcrumbs.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/ContextBreadcrumbs.js @@ -24,6 +24,8 @@ qx.Class.define("osparc.dashboard.ContextBreadcrumbs", { this._setLayout(new qx.ui.layout.HBox(5).set({ alignY: "middle" })); + + osparc.store.Store.getInstance().addListener("changeStudyBrowserContext", () => this.__rebuild(), this); }, events: { @@ -31,14 +33,6 @@ qx.Class.define("osparc.dashboard.ContextBreadcrumbs", { }, properties: { - currentContext: { - check: ["studiesAndFolders", "workspaces", "search"], - nullable: false, - init: "studiesAndFolders", - event: "changeCurrentContext", - apply: "__rebuild" - }, - currentWorkspaceId: { check: "Number", nullable: true, @@ -60,7 +54,8 @@ qx.Class.define("osparc.dashboard.ContextBreadcrumbs", { __rebuild: function() { this._removeAll(); - if (this.getCurrentContext() !== "studiesAndFolders") { + const currentContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + if (currentContext !== "studiesAndFolders") { return; } diff --git a/services/static-webserver/client/source/class/osparc/dashboard/FolderButtonItem.js b/services/static-webserver/client/source/class/osparc/dashboard/FolderButtonItem.js index 2932a774c92..ac919b73579 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/FolderButtonItem.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/FolderButtonItem.js @@ -46,6 +46,8 @@ qx.Class.define("osparc.dashboard.FolderButtonItem", { "folderSelected": "qx.event.type.Data", "folderUpdated": "qx.event.type.Data", "moveFolderToRequested": "qx.event.type.Data", + "trashFolderRequested": "qx.event.type.Data", + "untrashFolderRequested": "qx.event.type.Data", "deleteFolderRequested": "qx.event.type.Data", "changeContext": "qx.event.type.Data", }, @@ -85,7 +87,7 @@ qx.Class.define("osparc.dashboard.FolderButtonItem", { check: "Date", nullable: true, apply: "__applyLastModified" - } + }, }, members: { @@ -219,6 +221,16 @@ qx.Class.define("osparc.dashboard.FolderButtonItem", { menu.addSeparator(); + const trashButton = new qx.ui.menu.Button(this.tr("Trash"), "@FontAwesome5Solid/trash/12"); + trashButton.addListener("execute", () => this.__trashFolderRequested(), this); + menu.add(trashButton); + } else if (studyBrowserContext === "trash") { + const restoreButton = new qx.ui.menu.Button(this.tr("Restore"), "@MaterialIcons/restore_from_trash/16"); + restoreButton.addListener("execute", () => this.fireDataEvent("untrashFolderRequested", this.getFolder()), this); + menu.add(restoreButton); + + menu.addSeparator(); + const deleteButton = new qx.ui.menu.Button(this.tr("Delete"), "@FontAwesome5Solid/trash/12"); osparc.utils.Utils.setIdToWidget(deleteButton, "deleteFolderMenuItem"); deleteButton.addListener("execute", () => this.__deleteFolderRequested(), this); @@ -229,7 +241,9 @@ qx.Class.define("osparc.dashboard.FolderButtonItem", { }, __itemSelected: function(newVal) { - if (newVal) { + const studyBrowserContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + // do not allow selecting workspace + if (studyBrowserContext !== "trash" && newVal) { this.fireDataEvent("folderSelected", this.getFolderId()); } this.setValue(false); @@ -262,6 +276,24 @@ qx.Class.define("osparc.dashboard.FolderButtonItem", { folderEditor.addListener("cancel", () => win.close()); }, + __trashFolderRequested: function() { + const trashDays = osparc.store.StaticInfo.getInstance().getTrashRetentionDays(); + let msg = this.tr("Are you sure you want to move the Folder and all its content to the trash?"); + msg += "

" + this.tr("It will be permanently deleted after ") + trashDays + " days."; + const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ + caption: this.tr("Move to Trash"), + confirmText: this.tr("Move to Trash"), + confirmAction: "delete" + }); + confirmationWin.center(); + confirmationWin.open(); + confirmationWin.addListener("close", () => { + if (confirmationWin.getConfirmed()) { + this.fireDataEvent("trashFolderRequested", this.getFolderId()); + } + }, this); + }, + __deleteFolderRequested: function() { const msg = this.tr("Are you sure you want to delete") + " " + this.getTitle() + "?"; const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ diff --git a/services/static-webserver/client/source/class/osparc/dashboard/GridButtonItem.js b/services/static-webserver/client/source/class/osparc/dashboard/GridButtonItem.js index 828a0c74ba7..e9019262342 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/GridButtonItem.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/GridButtonItem.js @@ -188,8 +188,7 @@ qx.Class.define("osparc.dashboard.GridButtonItem", { }, __itemSelected: function() { - // It could be blocked by IN_USE or UNKNOWN_SERVICE - if (this.getBlocked() === true) { + if (this.isItemNotClickable()) { this.setValue(false); return; } diff --git a/services/static-webserver/client/source/class/osparc/dashboard/ListButtonItem.js b/services/static-webserver/client/source/class/osparc/dashboard/ListButtonItem.js index 71f59b970df..9c433550185 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/ListButtonItem.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/ListButtonItem.js @@ -274,8 +274,7 @@ qx.Class.define("osparc.dashboard.ListButtonItem", { }, __itemSelected: function() { - // It could be blocked by IN_USE or UNKNOWN_SERVICE - if (this.getBlocked() === true) { + if (this.isItemNotClickable()) { this.setValue(false); return; } diff --git a/services/static-webserver/client/source/class/osparc/dashboard/ResourceBrowserBase.js b/services/static-webserver/client/source/class/osparc/dashboard/ResourceBrowserBase.js index ca565c756f3..c007ca05f7e 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/ResourceBrowserBase.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/ResourceBrowserBase.js @@ -244,7 +244,7 @@ qx.Class.define("osparc.dashboard.ResourceBrowserBase", { this._addToLayout(searchBarFilter); }, - _createResourcesLayout: function() { + _createResourcesLayout: function(flatListId) { const toolbar = this._toolbar = new qx.ui.toolbar.ToolBar().set({ backgroundColor: "transparent", spacing: 10, @@ -255,7 +255,13 @@ qx.Class.define("osparc.dashboard.ResourceBrowserBase", { this.__viewModeLayout = new qx.ui.toolbar.Part(); - const resourcesContainer = this._resourcesContainer = new osparc.dashboard.ResourceContainerManager(); + const resourcesContainer = this._resourcesContainer = new osparc.dashboard.ResourceContainerManager(this._resourceType); + if (flatListId) { + const list = this._resourcesContainer.getFlatList(); + if (list) { + osparc.utils.Utils.setIdToWidget(list, flatListId); + } + } if (this._resourceType === "study") { const viewMode = osparc.utils.Utils.localCache.getLocalStorageItem("studiesViewMode"); if (viewMode) { @@ -270,6 +276,8 @@ qx.Class.define("osparc.dashboard.ResourceBrowserBase", { resourcesContainer.addListener("emptyStudyClicked", e => this._deleteResourceRequested(e.getData())); resourcesContainer.addListener("folderUpdated", e => this._folderUpdated(e.getData())); resourcesContainer.addListener("moveFolderToRequested", e => this._moveFolderToRequested(e.getData())); + resourcesContainer.addListener("trashFolderRequested", e => this._trashFolderRequested(e.getData())); + resourcesContainer.addListener("untrashFolderRequested", e => this._untrashFolderRequested(e.getData())); resourcesContainer.addListener("deleteFolderRequested", e => this._deleteFolderRequested(e.getData())); resourcesContainer.addListener("folderSelected", e => { const folderId = e.getData(); @@ -288,6 +296,8 @@ qx.Class.define("osparc.dashboard.ResourceBrowserBase", { this._changeContext(context, workspaceId, folderId); }, this); resourcesContainer.addListener("workspaceUpdated", e => this._workspaceUpdated(e.getData())); + resourcesContainer.addListener("trashWorkspaceRequested", e => this._trashWorkspaceRequested(e.getData())); + resourcesContainer.addListener("untrashWorkspaceRequested", e => this._untrashWorkspaceRequested(e.getData())); resourcesContainer.addListener("deleteWorkspaceRequested", e => this._deleteWorkspaceRequested(e.getData())); this._addToLayout(resourcesContainer); @@ -502,6 +512,14 @@ qx.Class.define("osparc.dashboard.ResourceBrowserBase", { throw new Error("Abstract method called!"); }, + _trashFolderRequested: function(folderId) { + throw new Error("Abstract method called!"); + }, + + _untrashFolderRequested: function(folder) { + throw new Error("Abstract method called!"); + }, + _deleteFolderRequested: function(folderId) { throw new Error("Abstract method called!"); }, diff --git a/services/static-webserver/client/source/class/osparc/dashboard/ResourceContainerManager.js b/services/static-webserver/client/source/class/osparc/dashboard/ResourceContainerManager.js index ba1485f024f..b33d5015231 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/ResourceContainerManager.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/ResourceContainerManager.js @@ -18,7 +18,7 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { extend: qx.ui.core.Widget, - construct: function() { + construct: function(resourceType) { this.base(arguments); this._setLayout(new qx.ui.layout.VBox(15)); @@ -27,19 +27,20 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { paddingBottom: 60 }); - this.__foldersList = []; this.__workspacesList = []; + this.__foldersList = []; this.__resourcesList = []; this.__groupedContainersList = []; + if (resourceType === "study") { + const workspacesContainer = this.__workspacesContainer = new osparc.dashboard.ToggleButtonContainer(); + this._add(workspacesContainer); + workspacesContainer.setVisibility(osparc.utils.DisabledPlugins.isFoldersEnabled() ? "visible" : "excluded"); - const workspacesContainer = this.__workspacesContainer = new osparc.dashboard.ToggleButtonContainer(); - workspacesContainer.setVisibility(osparc.utils.DisabledPlugins.isFoldersEnabled() ? "visible" : "excluded"); - - - const foldersContainer = this.__foldersContainer = new osparc.dashboard.ToggleButtonContainer(); - this._add(foldersContainer); - foldersContainer.setVisibility(osparc.utils.DisabledPlugins.isFoldersEnabled() ? "visible" : "excluded"); + const foldersContainer = this.__foldersContainer = new osparc.dashboard.ToggleButtonContainer(); + this._add(foldersContainer); + foldersContainer.setVisibility(osparc.utils.DisabledPlugins.isFoldersEnabled() ? "visible" : "excluded"); + } const nonGroupedContainer = this.__nonGroupedContainer = this.__createFlatList(); this._add(nonGroupedContainer); @@ -75,9 +76,13 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { "folderSelected": "qx.event.type.Data", "folderUpdated": "qx.event.type.Data", "moveFolderToRequested": "qx.event.type.Data", + "trashFolderRequested": "qx.event.type.Data", + "untrashFolderRequested": "qx.event.type.Data", "deleteFolderRequested": "qx.event.type.Data", "workspaceSelected": "qx.event.type.Data", "workspaceUpdated": "qx.event.type.Data", + "trashWorkspaceRequested": "qx.event.type.Data", + "untrashWorkspaceRequested": "qx.event.type.Data", "deleteWorkspaceRequested": "qx.event.type.Data", "changeContext": "qx.event.type.Data", }, @@ -260,9 +265,13 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { }, __cleanAll: function() { - if (this.__workspacesContainer) { - this.__workspacesContainer.removeAll(); + if (this._getChildren().includes(this.__nonGroupedContainer)) { + this._remove(this.__nonGroupedContainer); + } + if (this._getChildren().includes(this.__groupedContainers)) { + this._remove(this.__groupedContainers); } + if (this.__nonGroupedContainer) { this.__nonGroupedContainer.removeAll(); this.__nonGroupedContainer = null; @@ -274,7 +283,6 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { groupedContainer.getContentContainer().removeAll(); }); this.__groupedContainersList = []; - this._removeAll(); }, __addFoldersContainer: function() { @@ -299,9 +307,6 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { __rebuildLayout: function(resourceType) { this.__cleanAll(); - if (osparc.utils.DisabledPlugins.isFoldersEnabled()) { - this.__addFoldersContainer(); - } if (this.getGroupBy()) { const noGroupContainer = this.__createGroupContainer("no-group", "No Group", "transparent"); this.__groupedContainers.add(noGroupContainer); @@ -361,8 +366,9 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { }, reloadWorkspaces: function() { - this.__cleanAll(); - this._add(this.__workspacesContainer); + if (this.__workspacesContainer) { + this.__workspacesContainer.removeAll(); + } let workspacesCards = []; this.__workspacesList.forEach(workspaceData => workspacesCards.push(this.__workspaceToCard(workspaceData))); return workspacesCards; @@ -383,6 +389,8 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { [ "workspaceSelected", "workspaceUpdated", + "trashWorkspaceRequested", + "untrashWorkspaceRequested", "deleteWorkspaceRequested", ].forEach(eName => card.addListener(eName, e => this.fireDataEvent(eName, e.getData()))); return card; @@ -419,6 +427,8 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { "folderSelected", "folderUpdated", "moveFolderToRequested", + "trashFolderRequested", + "untrashFolderRequested", "deleteFolderRequested", "changeContext", ].forEach(eName => card.addListener(eName, e => this.fireDataEvent(eName, e.getData()))); @@ -494,7 +504,7 @@ qx.Class.define("osparc.dashboard.ResourceContainerManager", { } else { groupContainer.exclude(); } - this._add(groupContainer); + this.__groupedContainers.add(groupContainer); this.__moveNoGroupToLast(); } const card = this.__createCard(resourceData); diff --git a/services/static-webserver/client/source/class/osparc/dashboard/ResourceFilter.js b/services/static-webserver/client/source/class/osparc/dashboard/ResourceFilter.js index 0c452e3e33a..dcda538841c 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/ResourceFilter.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/ResourceFilter.js @@ -34,6 +34,7 @@ qx.Class.define("osparc.dashboard.ResourceFilter", { }, events: { + "trashContext": "qx.event.type.Event", "changeSharedWith": "qx.event.type.Data", "changeSelectedTags": "qx.event.type.Data", "changeServiceType": "qx.event.type.Data" @@ -42,6 +43,7 @@ qx.Class.define("osparc.dashboard.ResourceFilter", { members: { __resourceType: null, __workspacesAndFoldersTree: null, + __trashButton: null, __sharedWithButtons: null, __tagButtons: null, __serviceTypeButtons: null, @@ -49,6 +51,7 @@ qx.Class.define("osparc.dashboard.ResourceFilter", { __buildLayout: function() { if (this.__resourceType === "study" && osparc.utils.DisabledPlugins.isFoldersEnabled()) { this._add(this.__createWorkspacesAndFoldersTree()); + this._add(this.__createTrashBin()); } else { this._add(this.__createSharedWithFilterLayout()); } @@ -62,6 +65,16 @@ qx.Class.define("osparc.dashboard.ResourceFilter", { } }, + contextChanged: function(context, workspaceId, folderId) { + this.__workspacesAndFoldersTree.set({ + currentWorkspaceId: workspaceId, + currentFolderId: folderId, + }); + this.__workspacesAndFoldersTree.contextChanged(context); + + this.__trashButton.setValue(context === "trash"); + }, + /* WORKSPACES AND FOLDERS */ __createWorkspacesAndFoldersTree: function() { const workspacesAndFoldersTree = this.__workspacesAndFoldersTree = new osparc.dashboard.WorkspacesAndFoldersTree(); @@ -85,6 +98,76 @@ qx.Class.define("osparc.dashboard.ResourceFilter", { }, /* /WORKSPACES AND FOLDERS */ + /* TRASH BIN */ + __createTrashBin: function() { + const trashButton = this.__trashButton = new qx.ui.toolbar.RadioButton().set({ + value: false, + appearance: "filter-toggle-button", + label: this.tr("Trash"), + icon: "@FontAwesome5Solid/trash/18", + }); + trashButton.addListener("changeValue", e => { + const trashEnabled = e.getData(); + if (trashEnabled) { + this.fireEvent("trashContext"); + } + }); + this.evaluateTrashEmpty(); + return trashButton; + }, + + evaluateTrashEmpty: function() { + const studiesParams = { + url: { + offset: 0, + limit: 1, // just one + orderBy: JSON.stringify({ + field: "last_change_date", + direction: "desc" + }), + } + }; + const foldersParams = { + url: { + offset: 0, + limit: 1, // just one + orderBy: JSON.stringify({ + field: "modified_at", + direction: "desc" + }), + } + }; + const workspacesParams = { + url: { + offset: 0, + limit: 1, // just one + orderBy: JSON.stringify({ + field: "modified_at", + direction: "desc" + }), + } + }; + Promise.all([ + osparc.data.Resources.fetch("studies", "getPageTrashed", studiesParams), + osparc.data.Resources.fetch("folders", "getPageTrashed", foldersParams), + osparc.data.Resources.fetch("workspaces", "getPageTrashed", workspacesParams), + ]) + .then(values => { + const nTrashedStudies = values[0].length; + const nTrashedFolders = values[1].length; + const nTrashedWorkspaces = values[2].length; + this.setTrashEmpty((nTrashedStudies+nTrashedFolders+nTrashedWorkspaces) === 0); + }) + .catch(err => console.error(err)); + }, + + setTrashEmpty: function(isEmpty) { + this.__trashButton.set({ + textColor: isEmpty ? "text" : "danger-red" + }); + }, + /* /TRASH BIN */ + /* SHARED WITH */ __createSharedWithFilterLayout: function() { const sharedWithLayout = new qx.ui.container.Composite(new qx.ui.layout.VBox(5)); diff --git a/services/static-webserver/client/source/class/osparc/dashboard/ServiceBrowser.js b/services/static-webserver/client/source/class/osparc/dashboard/ServiceBrowser.js index 15278743e54..5fbaa4ebaf7 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/ServiceBrowser.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/ServiceBrowser.js @@ -129,11 +129,7 @@ qx.Class.define("osparc.dashboard.ServiceBrowser", { // LAYOUT // _createLayout: function() { this._createSearchBar(); - this._createResourcesLayout(); - const list = this._resourcesContainer.getFlatList(); - if (list) { - osparc.utils.Utils.setIdToWidget(list, "servicesList"); - } + this._createResourcesLayout("servicesList"); this.__addNewServiceButtons(); this._toolbar.add(new qx.ui.core.Spacer(), { diff --git a/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowser.js b/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowser.js index 882848dd295..c7ef8f916f2 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowser.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowser.js @@ -42,6 +42,9 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { construct: function() { this._resourceType = "study"; this.base(arguments); + + const store = osparc.store.Store.getInstance(); + this.bind("currentContext", store, "studyBrowserContext"); }, events: { @@ -50,7 +53,7 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { properties: { currentContext: { - check: ["studiesAndFolders", "workspaces", "search"], + check: ["studiesAndFolders", "workspaces", "search", "trash"], nullable: false, init: "studiesAndFolders", event: "changeCurrentContext" @@ -96,6 +99,7 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { __workspacesList: null, __foldersList: null, __loadingFolders: null, + __loadingWorkspaces: null, // overridden initResources: function() { @@ -160,10 +164,48 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { }, __reloadWorkspaces: function() { + if ( + !osparc.auth.Manager.getInstance().isLoggedIn() || + !osparc.utils.DisabledPlugins.isFoldersEnabled() || + this.getCurrentContext() === "studiesAndFolders" || + this.getCurrentContext() === "search" || // not yet implemented for workspaces + this.__loadingWorkspaces + ) { + return; + } + + let request = null; + switch (this.getCurrentContext()) { + case "search": { + const filterData = this._searchBarFilter.getFilterData(); + const text = filterData.text ? encodeURIComponent(filterData.text) : ""; + request = osparc.store.Workspaces.getInstance().searchWorkspaces(text); + break; + } + case "workspaces": { + request = osparc.store.Workspaces.getInstance().fetchWorkspaces(); + break; + } + case "trash": + request = osparc.store.Workspaces.getInstance().fetchAllTrashedWorkspaces(); + break; + } + + this.__loadingWorkspaces = true; this.__setWorkspacesToList([]); - osparc.store.Workspaces.getInstance().fetchWorkspaces() + request .then(workspaces => { this.__setWorkspacesToList(workspaces); + if (this.getCurrentContext() === "trash") { + if (workspaces.length) { + this.__header.getChildControl("empty-trash-button").show(); + } + } + }) + .catch(console.error) + .finally(() => { + this.__addNewWorkspaceButton(); + this.__loadingWorkspaces = null; }); }, @@ -177,7 +219,6 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { return; } - this.__loadingFolders = true; let request = null; switch (this.getCurrentContext()) { case "search": { @@ -192,14 +233,27 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { request = osparc.store.Folders.getInstance().fetchFolders(folderId, workspaceId, this.getOrderBy()); break; } + case "trash": + request = osparc.store.Folders.getInstance().fetchAllTrashedFolders(this.getOrderBy()); + break; } + + this.__loadingFolders = true; this.__setFoldersToList([]); request .then(folders => { this.__setFoldersToList(folders); + if (this.getCurrentContext() === "trash") { + if (folders.length) { + this.__header.getChildControl("empty-trash-button").show(); + } + } }) .catch(console.error) - .finally(() => this.__loadingFolders = null); + .finally(() => { + this.__addNewFolderButton(); + this.__loadingFolders = null; + }); }, __reloadStudies: function() { @@ -257,6 +311,12 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { this._resourcesContainer.getFlatList().nextRequest = resp["_links"]["next"]; } + if (this.getCurrentContext() === "trash") { + if (this._resourcesList.length) { + this.__header.getChildControl("empty-trash-button").show(); + } + } + // Show Quick Start if there are no studies in the root folder of the personal workspace const quickStartInfo = osparc.product.quickStart.Utils.getQuickStart(); if (quickStartInfo) { @@ -267,6 +327,7 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { const nStudies = "_meta" in resp ? resp["_meta"]["total"] : 0; if ( nStudies === 0 && + this.getCurrentContext() === "studiesAndFolders" && this.getCurrentWorkspaceId() === null && this.getCurrentFolderId() === null ) { @@ -354,9 +415,6 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { const cards = this._resourcesContainer.reloadCards("studies"); this.__configureStudyCards(cards); - // they were removed in the above reloadCards - this.__reloadFolders(); - this.__addNewStudyButtons(); const loadMoreBtn = this.__createLoadMoreButton(); @@ -382,6 +440,12 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { __reloadWorkspaceCards: function() { this._resourcesContainer.setWorkspacesToList(this.__workspacesList); this._resourcesContainer.reloadWorkspaces(); + }, + + __addNewWorkspaceButton: function() { + if (this.getCurrentContext() !== "workspaces") { + return; + } const newWorkspaceCard = new osparc.dashboard.WorkspaceButtonNew(); newWorkspaceCard.setCardKey("new-workspace"); @@ -401,13 +465,44 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { }, _workspaceUpdated: function() { - this.__reloadWorkspaceCards(); + this.__reloadWorkspaces(); + }, + + _trashWorkspaceRequested: function(workspaceId) { + osparc.store.Workspaces.getInstance().trashWorkspace(workspaceId) + .then(() => { + this.__reloadWorkspaces(); + const msg = this.tr("Successfully moved to Trash"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.setTrashEmpty(false); + }) + .catch(err => { + console.error(err); + osparc.FlashMessenger.getInstance().logAs(err, "ERROR"); + }); + }, + + _untrashWorkspaceRequested: function(workspace) { + osparc.store.Workspaces.getInstance().untrashWorkspace(workspace) + .then(() => { + this.__reloadWorkspaces(); + const msg = this.tr("Successfully restored"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.evaluateTrashEmpty(); + }) + .catch(err => { + console.error(err); + osparc.FlashMessenger.getInstance().logAs(err, "ERROR"); + }); }, _deleteWorkspaceRequested: function(workspaceId) { osparc.store.Workspaces.getInstance().deleteWorkspace(workspaceId) .then(() => { this.__reloadWorkspaces(); + const msg = this.tr("Successfully deleted"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.evaluateTrashEmpty(); }) .catch(err => { console.error(err); @@ -420,8 +515,6 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { __reloadFolderCards: function() { this._resourcesContainer.setFoldersToList(this.__foldersList); this._resourcesContainer.reloadFolders(); - - this.__addNewFolderButton(); }, __addNewFolderButton: function() { @@ -544,9 +637,42 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { .catch(err => console.error(err)); }, + _trashFolderRequested: function(folderId) { + osparc.store.Folders.getInstance().trashFolder(folderId, this.getCurrentWorkspaceId()) + .then(() => { + this.__reloadFolders(); + const msg = this.tr("Successfully moved to Trash"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.setTrashEmpty(false); + }) + .catch(err => { + console.error(err); + osparc.FlashMessenger.getInstance().logAs(err, "ERROR"); + }) + }, + + _untrashFolderRequested: function(folder) { + osparc.store.Folders.getInstance().untrashFolder(folder) + .then(() => { + this.__reloadFolders(); + const msg = this.tr("Successfully restored"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.evaluateTrashEmpty(); + }) + .catch(err => { + console.error(err); + osparc.FlashMessenger.getInstance().logAs(err, "ERROR"); + }) + }, + _deleteFolderRequested: function(folderId) { osparc.store.Folders.getInstance().deleteFolder(folderId, this.getCurrentWorkspaceId()) - .then(() => this.__reloadFolders()) + .then(() => { + this.__reloadFolders(); + const msg = this.tr("Successfully deleted"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.evaluateTrashEmpty(); + }) .catch(err => console.error(err)); }, // /FOLDERS @@ -555,7 +681,7 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { cards.forEach(card => { card.setMultiSelectionMode(this.getMultiSelection()); card.addListener("tap", e => { - if (card.getBlocked() === true) { + if (card.isItemNotClickable()) { card.setValue(false); } else { this.__itemClicked(card, e.getNativeEvent().shiftKey); @@ -634,6 +760,11 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { // loose equality: will do a Number to String conversion if necessary sameContext &= key in reqParams && reqParams[key] == value; }); + // both ways + Object.entries(reqParams).forEach(([key, value]) => { + // loose equality: will do a Number to String conversion if necessary + sameContext &= key in currentParams && currentParams[key] == value; + }); return !sameContext; }, @@ -711,6 +842,9 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { let request = null; switch (this.getCurrentContext()) { + case "trash": + request = osparc.data.Resources.fetch("studies", "getPageTrashed", params, options); + break; case "search": request = osparc.data.Resources.fetch("studies", "getPageSearch", params, options); break; @@ -864,15 +998,11 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { if (osparc.utils.DisabledPlugins.isFoldersEnabled()) { const header = this.__header = new osparc.dashboard.StudyBrowserHeader(); + this.__header.addListener("emptyTrashRequested", () => this.__emptyTrash(), this); this._addToLayout(header); } - this._createResourcesLayout(); - - const list = this._resourcesContainer.getFlatList(); - if (list) { - osparc.utils.Utils.setIdToWidget(list, "studiesList"); - } + this._createResourcesLayout("studiesList"); const importStudyButton = this.__createImportButton(); const isDisabled = osparc.utils.DisabledPlugins.isImportDisabled(); @@ -882,10 +1012,13 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { const selectStudiesButton = this.__createSelectButton(); this._toolbar.add(selectStudiesButton); - const studiesMoveButton = this.__createMoveStudiesButton(false); + const studiesMoveButton = this.__createMoveStudiesButton(); this._toolbar.add(studiesMoveButton); - const studiesDeleteButton = this.__createDeleteButton(false); + const studiesTrashButton = this.__createTrashStudiesButton(); + this._toolbar.add(studiesTrashButton); + + const studiesDeleteButton = this.__createDeleteStudiesButton(); this._toolbar.add(studiesDeleteButton); this._toolbar.add(new qx.ui.core.Spacer(), { @@ -924,8 +1057,13 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { label: selection.length > 1 ? this.tr("Move selected")+" ("+selection.length+")" : this.tr("Move") }); + studiesTrashButton.set({ + visibility: selection.length && currentContext === "studiesAndFolders" ? "visible" : "excluded", + label: selection.length > 1 ? this.tr("Trash selected")+" ("+selection.length+")" : this.tr("Trash") + }); + studiesDeleteButton.set({ - visibility: selection.length ? "visible" : "excluded", + visibility: selection.length && currentContext === "trash" ? "visible" : "excluded", label: selection.length > 1 ? this.tr("Delete selected")+" ("+selection.length+")" : this.tr("Delete") }); }); @@ -956,6 +1094,10 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { } }, this); + this._resourceFilter.addListener("trashContext", () => { + this._changeContext("trash"); + }); + this._searchBarFilter.addListener("filterChanged", e => { const filterData = e.getData(); if (filterData.text) { @@ -987,41 +1129,56 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { currentWorkspaceId: workspaceId, currentFolderId: folderId, }); - this._loadingResourcesBtn.setFetching(false); this.resetSelection(); this.setMultiSelection(false); - this.invalidateStudies(); - this._resourcesContainer.setResourcesToList([]); + + // reset lists + this.__setWorkspacesToList([]); + this.__setFoldersToList([]); + this._resourcesList = []; + this._resourcesContainer.setResourcesToList(this._resourcesList); + this._resourcesContainer.reloadCards("studies"); this._toolbar.show(); - if (context === "search") { - this.__reloadFolders(); - this.__reloadStudies(); - } else if (context === "workspaces") { - this._toolbar.hide(); - this._searchBarFilter.resetFilters(); - this.__reloadWorkspaces(); - } else if (context === "studiesAndFolders") { - this._searchBarFilter.resetFilters(); - this.__reloadFolders(); - this.__reloadStudies(); + switch (this.getCurrentContext()) { + case "studiesAndFolders": + this._searchBarFilter.resetFilters(); + this.__reloadFolders(); + this._loadingResourcesBtn.setFetching(false); + this.invalidateStudies(); + this.__reloadStudies(); + break; + case "workspaces": + this._toolbar.exclude(); + this._searchBarFilter.resetFilters(); + this.__reloadWorkspaces(); + break; + case "search": + this.__reloadWorkspaces(); + this.__reloadFolders(); + this._loadingResourcesBtn.setFetching(false); + this.invalidateStudies(); + this.__reloadStudies(); + break; + case "trash": + this._searchBarFilter.resetFilters(); + this.__reloadWorkspaces(); + this.__reloadFolders(); + this._loadingResourcesBtn.setFetching(false); + this.invalidateStudies(); + this.__reloadStudies(); + break; } // notify header const header = this.__header; header.set({ - currentContext: context, currentWorkspaceId: workspaceId, currentFolderId: folderId, }); - // notify workspacesAndFoldersTree - const workspacesAndFoldersTree = this._resourceFilter.getWorkspacesAndFoldersTree(); - workspacesAndFoldersTree.set({ - currentWorkspaceId: workspaceId, - currentFolderId: folderId, - }); - workspacesAndFoldersTree.contextChanged(context); + // notify Filters on the left + this._resourceFilter.contextChanged(context, workspaceId, folderId); } }, @@ -1129,7 +1286,32 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { return moveStudiesButton; }, - __createDeleteButton: function() { + __createTrashStudiesButton: function() { + const trashButton = new qx.ui.form.Button(this.tr("Trash"), "@FontAwesome5Solid/trash/14").set({ + appearance: "danger-button", + visibility: "excluded" + }); + osparc.utils.Utils.setIdToWidget(trashButton, "deleteStudiesBtn"); + trashButton.addListener("execute", () => { + const selection = this._resourcesContainer.getSelection(); + const preferencesSettings = osparc.Preferences.getInstance(); + if (preferencesSettings.getConfirmDeleteStudy()) { + const win = this.__createConfirmTrashWindow(selection.map(button => button.getTitle())); + win.center(); + win.open(); + win.addListener("close", () => { + if (win.getConfirmed()) { + this.__trashStudies(selection.map(button => this.__getStudyData(button.getUuid(), false)), false); + } + }, this); + } else { + this.__trashStudies(selection.map(button => this.__getStudyData(button.getUuid(), false)), false); + } + }, this); + return trashButton; + }, + + __createDeleteStudiesButton: function() { const deleteButton = new qx.ui.form.Button(this.tr("Delete"), "@FontAwesome5Solid/trash/14").set({ appearance: "danger-button", visibility: "excluded" @@ -1139,7 +1321,7 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { const selection = this._resourcesContainer.getSelection(); const preferencesSettings = osparc.Preferences.getInstance(); if (preferencesSettings.getConfirmDeleteStudy()) { - const win = this.__createConfirmWindow(selection.map(button => button.getTitle())); + const win = this.__createConfirmDeleteWindow(selection.map(button => button.getTitle())); win.center(); win.open(); win.addListener("close", () => { @@ -1154,6 +1336,20 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { return deleteButton; }, + __emptyTrash: function() { + const win = this.__createConfirmEmptyTrashWindow(); + win.center(); + win.open(); + win.addListener("close", () => { + if (win.getConfirmed()) { + osparc.data.Resources.fetch("trash", "delete") + .then(() => { + this.__resetStudiesList(); + }); + } + }, this); + }, + __createSelectButton: function() { const selectButton = new qx.ui.form.ToggleButton().set({ appearance: "form-button-outlined", @@ -1297,9 +1493,25 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { const menu = card.getMenu(); const studyData = card.getResourceData(); + const trashed = Boolean(studyData["trashedAt"]); const writeAccess = osparc.data.model.Study.canIWrite(studyData["accessRights"]); const deleteAccess = osparc.data.model.Study.canIDelete(studyData["accessRights"]); + if (this.getCurrentContext() === "trash") { + if (trashed) { + if (writeAccess) { + const untrashButton = this.__getUntrashStudyMenuButton(studyData); + menu.add(untrashButton); + } + if (deleteAccess) { + const deleteButton = this.__getDeleteStudyMenuButton(studyData, false); + menu.addSeparator(); + menu.add(deleteButton); + } + } + return; + } + const openButton = this._getOpenMenuButton(studyData); if (openButton) { menu.add(openButton); @@ -1363,11 +1575,14 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { } if (deleteAccess) { + menu.addSeparator(); + const trashButton = this.__getTrashStudyMenuButton(studyData, false); + menu.add(trashButton); + } else if (this.__deleteOrRemoveMe(studyData) === "remove") { + // if I'm a collaborator, let me remove myself from the study. In that case it would be a Delete for me + menu.addSeparator(); const deleteButton = this.__getDeleteStudyMenuButton(studyData, false); - if (deleteButton) { - menu.addSeparator(); - menu.add(deleteButton); - } + menu.add(deleteButton); } card.evaluateMenuButtons(); @@ -1549,13 +1764,33 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { }, _deleteResourceRequested: function(studyId) { - this.__deleteStudyRequested(this.__getStudyData(studyId)); + if (this.getCurrentContext() === "trash") { + this.__deleteStudyRequested(this.__getStudyData(studyId)); + } else { + this.__trashStudyRequested(this.__getStudyData(studyId)); + } + }, + + __trashStudyRequested: function(studyData) { + const preferencesSettings = osparc.Preferences.getInstance(); + if (preferencesSettings.getConfirmDeleteStudy()) { + const win = this.__createConfirmTrashWindow([studyData.name]); + win.center(); + win.open(); + win.addListener("close", () => { + if (win.getConfirmed()) { + this.__trashStudy(studyData); + } + }, this); + } else { + this.__trashStudy(studyData); + } }, __deleteStudyRequested: function(studyData) { const preferencesSettings = osparc.Preferences.getInstance(); if (preferencesSettings.getConfirmDeleteStudy()) { - const win = this.__createConfirmWindow([studyData.name]); + const win = this.__deleteOrRemoveMe(studyData) === "remove" ? this.__createConfirmRemoveForMeWindow(studyData.name) : this.__createConfirmDeleteWindow([studyData.name]); win.center(); win.open(); win.addListener("close", () => { @@ -1568,6 +1803,27 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { } }, + __getTrashStudyMenuButton: function(studyData) { + const trashButton = new qx.ui.menu.Button(this.tr("Trash"), "@FontAwesome5Solid/trash/12"); + trashButton["trashButton"] = true; + trashButton.set({ + appearance: "menu-button" + }); + osparc.utils.Utils.setIdToWidget(trashButton, "studyItemMenuDelete"); + trashButton.addListener("execute", () => this.__trashStudyRequested(studyData), this); + return trashButton; + }, + + __getUntrashStudyMenuButton: function(studyData) { + const restoreButton = new qx.ui.menu.Button(this.tr("Restore"), "@MaterialIcons/restore_from_trash/16"); + restoreButton["untrashButton"] = true; + restoreButton.set({ + appearance: "menu-button" + }); + restoreButton.addListener("execute", () => this.__untrashStudy(studyData), this); + return restoreButton; + }, + __getDeleteStudyMenuButton: function(studyData) { const deleteButton = new qx.ui.menu.Button(this.tr("Delete"), "@FontAwesome5Solid/trash/12"); deleteButton["deleteButton"] = true; @@ -1730,17 +1986,60 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { req.send(body); }, - __doDeleteStudy: function(studyData) { + __untrashStudy: function(studyData) { + osparc.store.Store.getInstance().untrashStudy(studyData.uuid) + .then(() => { + this.__removeFromStudyList(studyData.uuid); + const msg = this.tr("Successfully restored"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.evaluateTrashEmpty(); + }) + .catch(err => { + console.error(err); + osparc.FlashMessenger.getInstance().logAs(err, "ERROR"); + }) + .finally(() => this.resetSelection()); + }, + + __trashStudy: function(studyData) { + osparc.store.Store.getInstance().trashStudy(studyData.uuid) + .then(() => { + this.__removeFromStudyList(studyData.uuid); + const msg = this.tr("Successfully moved to Trash"); + osparc.FlashMessenger.getInstance().logAs(msg, "INFO"); + this._resourceFilter.setTrashEmpty(false); + }) + .catch(err => { + console.error(err); + osparc.FlashMessenger.getInstance().logAs(err, "ERROR"); + }) + .finally(() => this.resetSelection()); + }, + + __trashStudies: function(studiesData) { + studiesData.forEach(studyData => this.__trashStudy(studyData)); + }, + + __deleteOrRemoveMe: function(studyData) { + const deleteAccess = osparc.data.model.Study.canIDelete(studyData["accessRights"]); const myGid = osparc.auth.Data.getInstance().getGroupId(); const collabGids = Object.keys(studyData["accessRights"]); const amICollaborator = collabGids.indexOf(myGid) > -1; + return (!deleteAccess && collabGids.length > 1 && amICollaborator) ? "remove" : "delete"; + }, + __removeMeFromCollaborators: function(studyData) { + const arCopy = osparc.utils.Utils.deepCloneObject(studyData["accessRights"]); + // remove me from collaborators + const myGid = osparc.auth.Data.getInstance().getGroupId(); + delete arCopy[myGid]; + return osparc.info.StudyUtils.patchStudyData(studyData, "accessRights", arCopy); + }, + + __doDeleteStudy: function(studyData) { let operationPromise = null; - if (collabGids.length > 1 && amICollaborator) { - const arCopy = osparc.utils.Utils.deepCloneObject(studyData["accessRights"]); - // remove collaborator - delete arCopy[myGid]; - operationPromise = osparc.info.StudyUtils.patchStudyData(studyData, "accessRights", arCopy); + if (this.__deleteOrRemoveMe(studyData) === "remove") { + operationPromise = this.__removeMeFromCollaborators(studyData); } else { // delete study operationPromise = osparc.store.Store.getInstance().deleteStudy(studyData.uuid); @@ -1758,10 +2057,41 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { studiesData.forEach(studyData => this.__doDeleteStudy(studyData)); }, - __createConfirmWindow: function(studyNames) { - const rUSure = this.tr("Are you sure you want to delete"); + __createConfirmTrashWindow: function(studyNames) { + let msg = this.tr("Are you sure you want to move"); + if (studyNames.length > 1) { + const studiesText = osparc.product.Utils.getStudyAlias({plural: true}); + msg += ` ${studyNames.length} ${studiesText} ` + } else { + msg += ` '${studyNames[0]}' `; + } + msg += this.tr("to the Trash?"); + const trashDays = osparc.store.StaticInfo.getInstance().getTrashRetentionDays(); + msg += "

" + (studyNames.length > 1 ? "They" : "It") + this.tr(` will be permanently deleted after ${trashDays} days.`); + const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ + caption: this.tr("Move to Trash"), + confirmText: this.tr("Move to Trash"), + confirmAction: "delete" + }); + osparc.utils.Utils.setIdToWidget(confirmationWin.getConfirmButton(), "confirmDeleteStudyBtn"); + return confirmationWin; + }, + + __createConfirmRemoveForMeWindow: function(studyName) { + const msg = `'${studyName} ` + this.tr("will be removed from your list. Collaborators will still have access."); + const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ + caption: this.tr("Remove"), + confirmText: this.tr("Remove"), + confirmAction: "delete" + }); + osparc.utils.Utils.setIdToWidget(confirmationWin.getConfirmButton(), "confirmDeleteStudyBtn"); + return confirmationWin; + }, + + __createConfirmDeleteWindow: function(studyNames) { + let msg = this.tr("Are you sure you want to delete"); const studyAlias = osparc.product.Utils.getStudyAlias({plural: studyNames.length > 1}); - const msg = rUSure + (studyNames.length > 1 ? ` ${studyNames.length} ${studyAlias}?` : ` ${studyNames[0]}?`) + msg += (studyNames.length > 1 ? ` ${studyNames.length} ${studyAlias}?` : ` ${studyNames[0]}?`); const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ caption: this.tr("Delete") + " " + studyAlias, confirmText: this.tr("Delete"), @@ -1771,6 +2101,16 @@ qx.Class.define("osparc.dashboard.StudyBrowser", { return confirmationWin; }, + __createConfirmEmptyTrashWindow: function() { + const msg = this.tr("Items in the bin will be permanently deleted"); + const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ + caption: this.tr("Delete"), + confirmText: this.tr("Delete forever"), + confirmAction: "delete" + }); + return confirmationWin; + }, + // TASKS // __tasksReceived: function(tasks) { tasks.forEach(taskData => this._taskDataReceived(taskData)); diff --git a/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowserHeader.js b/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowserHeader.js index 87a6a366b58..cb2e130cfcf 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowserHeader.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/StudyBrowserHeader.js @@ -16,7 +16,7 @@ ************************************************************************ */ /** - * Widget used for displaying a Workspace information + * Widget used for displaying a Study Browser's context information * */ @@ -35,27 +35,20 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { alignY: "middle", }); - this.__spacers = []; - this.initCurrentWorkspaceId(); this.initCurrentFolderId(); + + osparc.store.Store.getInstance().addListener("changeStudyBrowserContext", () => this.__buildLayout(), this); }, events: { "locationChanged": "qx.event.type.Data", "workspaceUpdated": "qx.event.type.Data", - "deleteWorkspaceRequested": "qx.event.type.Data" + "deleteWorkspaceRequested": "qx.event.type.Data", + "emptyTrashRequested": "qx.event.type.Event", }, properties: { - currentContext: { - check: ["studiesAndFolders", "workspaces", "search"], - nullable: false, - init: "studiesAndFolders", - event: "changeCurrentContext", - apply: "__buildLayout" - }, - currentWorkspaceId: { check: "Number", nullable: true, @@ -76,7 +69,7 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { nullable: false, init: {}, event: "changeAccessRights", - apply: "__applyAccessRights" + apply: "__updateShareInfo" }, myAccessRights: { @@ -89,7 +82,17 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { }, statics: { - HEIGHT: 36 + HEIGHT: 36, + POS: { + ICON: 0, + TITLE: 1, + BREADCRUMBS: 2, + EDIT_BUTTON: 3, + SHARE_LAYOUT: 4, + ROLE_LAYOUT: 5, + DESCRIPTION: 2, + EMPTY_TRASH_BUTTON: 3, + } }, members: { @@ -106,26 +109,25 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { allowGrowY: true, decorator: "rounded", }); - this._add(control); + this._addAt(control, this.self().POS.ICON); break; case "title": control = new qx.ui.basic.Label().set({ font: "text-16", alignY: "middle", }); - this._add(control); + this._addAt(control, this.self().POS.TITLE); break; case "breadcrumbs": control = new osparc.dashboard.ContextBreadcrumbs(); this.bind("currentWorkspaceId", control, "currentWorkspaceId"); this.bind("currentFolderId", control, "currentFolderId"); - this.bind("currentContext", control, "currentContext"); control.bind("currentWorkspaceId", this, "currentWorkspaceId"); control.bind("currentFolderId", this, "currentFolderId"); control.addListener("locationChanged", e => { this.fireDataEvent("locationChanged", e.getData()) }); - this._add(control); + this._addAt(control, this.self().POS.BREADCRUMBS); break; case "edit-button": control = new qx.ui.form.MenuButton().set({ @@ -142,14 +144,15 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { control.getContentElement().setStyles({ "border-radius": `${22 / 2}px` }); - this._add(control); + this._addAt(control, this.self().POS.EDIT_BUTTON); break; case "share-layout": - this.__addSpacer(); control = new qx.ui.container.Composite(new qx.ui.layout.HBox(10).set({ - alignY: "middle" - })); - this._add(control); + alignY: "middle", + })).set({ + marginLeft: 10, + }); + this._addAt(control, this.self().POS.SHARE_LAYOUT); break; case "share-text": { control = new qx.ui.basic.Label().set({ @@ -160,11 +163,12 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { break; } case "role-layout": - this.__addSpacer(); control = new qx.ui.container.Composite(new qx.ui.layout.HBox(5).set({ - alignY: "middle" - })); - this._add(control); + alignY: "middle", + })).set({ + marginLeft: 10, + }); + this._addAt(control, this.self().POS.ROLE_LAYOUT); break; case "role-text": { control = new qx.ui.basic.Label().set({ @@ -180,6 +184,24 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { layout.addAt(control, 1); break; } + case "description": { + control = new qx.ui.basic.Label().set({ + font: "text-14", + alignY: "middle", + }); + this._addAt(control, this.self().POS.DESCRIPTION); + break; + } + case "empty-trash-button": { + control = new qx.ui.form.Button(this.tr("Empty Trash"), "@FontAwesome5Solid/trash/14").set({ + appearance: "danger-button", + allowGrowY: false, + alignY: "middle", + }); + control.addListener("execute", () => this.fireEvent("emptyTrashRequested")); + this._addAt(control, this.self().POS.EMPTY_TRASH_BUTTON); + break; + } } return control || this.base(arguments, id); @@ -198,66 +220,82 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { __buildLayout: function() { this.getChildControl("icon"); const title = this.getChildControl("title"); - title.resetCursor(); - title.removeListener("tap", this.__titleTapped, this); - this.getChildControl("breadcrumbs"); - this.getChildControl("edit-button").exclude(); - this.resetAccessRights(); - this.resetMyAccessRights(); - - const currentContext = this.getCurrentContext(); - if (currentContext === "search") { - this.__setIcon("@FontAwesome5Solid/search/24"); - title.set({ - value: this.tr("Search results"), - }); - } else if (currentContext === "workspaces") { - this.__setIcon(osparc.store.Workspaces.iconPath(32)); - title.set({ - value: this.tr("Shared Workspaces"), - }) - } else if (currentContext === "studiesAndFolders") { - const workspaceId = this.getCurrentWorkspaceId(); - title.setCursor("pointer"); - title.addListener("tap", this.__titleTapped, this); - const workspace = osparc.store.Workspaces.getInstance().getWorkspace(workspaceId); - if (workspace) { - const thumbnail = workspace.getThumbnail(); - this.__setIcon(thumbnail ? thumbnail : osparc.store.Workspaces.iconPath(32)); - workspace.bind("name", title, "value"); - workspace.bind("accessRights", this, "accessRights"); - workspace.bind("myAccessRights", this, "myAccessRights"); - } else { - this.__setIcon("@FontAwesome5Solid/home/30"); - title.setValue(this.tr("My Workspace")); - } - } - }, - __addSpacer: function() { - const spacer = new qx.ui.basic.Label("-").set({ - font: "text-16", - alignY: "middle", + const locationBreadcrumbs = this.getChildControl("breadcrumbs").set({ + visibility: "excluded" + }); + const editWorkspace = this.getChildControl("edit-button").set({ + visibility: "excluded" + }); + const shareWorkspaceLayout = this.getChildControl("share-layout").set({ + visibility: "excluded" + }); + const roleWorkspaceLayout = this.getChildControl("role-layout").set({ + visibility: "excluded" + }); + + const description = this.getChildControl("description").set({ + visibility: "excluded" }); - this.__spacers.push(spacer); - this._add(spacer); + // the study browser will take care of making it visible + this.getChildControl("empty-trash-button").set({ + visibility: "excluded" + }); + + const currentContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + switch (currentContext) { + case "studiesAndFolders": { + const workspaceId = this.getCurrentWorkspaceId(); + title.setCursor("pointer"); + title.addListener("tap", this.__titleTapped, this); + locationBreadcrumbs.show(); + const workspace = osparc.store.Workspaces.getInstance().getWorkspace(workspaceId); + if (workspace) { + const thumbnail = workspace.getThumbnail(); + this.__setIcon(thumbnail ? thumbnail : osparc.store.Workspaces.iconPath(32)); + workspace.bind("name", title, "value"); + editWorkspace.show(); + shareWorkspaceLayout.show(); + roleWorkspaceLayout.show(); + workspace.bind("accessRights", this, "accessRights"); + workspace.bind("myAccessRights", this, "myAccessRights"); + } else { + this.__setIcon("@FontAwesome5Solid/home/30"); + title.setValue(this.tr("My Workspace")); + } + break; + } + case "workspaces": + this.__setIcon(osparc.store.Workspaces.iconPath(32)); + title.setValue(this.tr("Shared Workspaces")); + break; + case "search": + this.__setIcon("@FontAwesome5Solid/search/24"); + title.setValue(this.tr("Search results")); + break; + case "trash": { + this.__setIcon("@FontAwesome5Solid/trash/20"); + title.setValue(this.tr("Trash")); + const trashDays = osparc.store.StaticInfo.getInstance().getTrashRetentionDays(); + description.set({ + value: this.tr(`Items in the bin will be permanently deleted after ${trashDays} days.`), + visibility: "visible", + }); + break; + } + } }, - __resetIcon: function() { + __setIcon: function(source) { + // reset icon first const icon = this.getChildControl("icon"); const image = icon.getChildControl("image"); image.resetSource(); icon.getContentElement().setStyles({ "background-image": "none" }); - }, - __setIcon: function(source) { - this.__resetIcon(); - - const icon = this.getChildControl("icon"); if (source.includes("@")) { - const image = icon.getChildControl("image"); image.set({ source }); @@ -272,10 +310,6 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { } }, - __showSpacers: function(show) { - this.__spacers.forEach(spacer => spacer.setVisibility(show ? "visible" : "excluded")); - }, - __getShareIcon: function() { // reset previous const layout = this.getChildControl("share-layout"); @@ -292,7 +326,7 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { return shareIcon; }, - __applyAccessRights: function(accessRights) { + __updateShareInfo: function(accessRights) { const shareIcon = this.__getShareIcon(); const shareText = this.getChildControl("share-text"); if (accessRights && Object.keys(accessRights).length) { @@ -300,11 +334,9 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { shareText.setValue(Object.keys(accessRights).length + " members"); shareIcon.show(); shareText.show(); - this.__showSpacers(true); } else { shareIcon.exclude(); shareText.exclude(); - this.__showSpacers(false); } }, @@ -312,7 +344,8 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { const editButton = this.getChildControl("edit-button"); const roleText = this.getChildControl("role-text"); const roleIcon = this.getChildControl("role-icon"); - if (value && Object.keys(value).length) { + const currentContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + if (currentContext === "studiesAndFolders" && value && Object.keys(value).length) { editButton.setVisibility(value["delete"] ? "visible" : "excluded"); const menu = new qx.ui.menu.Menu().set({ position: "bottom-right" @@ -328,12 +361,10 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { roleText.setValue(osparc.data.Roles.WORKSPACE[val].label); roleText.show(); roleIcon.show(); - this.__showSpacers(true); } else { editButton.exclude(); roleText.exclude(); roleIcon.exclude(); - this.__showSpacers(false); } }, @@ -346,6 +377,7 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { win.close(); this.__buildLayout(); }, this); + workspaceEditor.addListener("cancel", () => win.close()); }, __openShareWith: function() { @@ -355,7 +387,7 @@ qx.Class.define("osparc.dashboard.StudyBrowserHeader", { const win = osparc.ui.window.Window.popUpInWindow(permissionsView, title, 500, 400); permissionsView.addListener("updateAccessRights", () => { win.close(); - this.__applyAccessRights(workspace.getAccessRights()); + this.__updateShareInfo(workspace.getAccessRights()); }, this); }, } diff --git a/services/static-webserver/client/source/class/osparc/dashboard/TemplateBrowser.js b/services/static-webserver/client/source/class/osparc/dashboard/TemplateBrowser.js index 3ff37cd8412..0b6fc8ccd26 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/TemplateBrowser.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/TemplateBrowser.js @@ -246,11 +246,7 @@ qx.Class.define("osparc.dashboard.TemplateBrowser", { // LAYOUT // _createLayout: function() { this._createSearchBar(); - this._createResourcesLayout(); - const list = this._resourcesContainer.getFlatList(); - if (list) { - osparc.utils.Utils.setIdToWidget(list, "templatesList"); - } + this._createResourcesLayout("templatesList"); const updateAllButton = this.__createUpdateAllButton(); if (updateAllButton) { diff --git a/services/static-webserver/client/source/class/osparc/dashboard/WorkspaceButtonItem.js b/services/static-webserver/client/source/class/osparc/dashboard/WorkspaceButtonItem.js index 187be3598e2..eb777ca5dd7 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/WorkspaceButtonItem.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/WorkspaceButtonItem.js @@ -45,7 +45,9 @@ qx.Class.define("osparc.dashboard.WorkspaceButtonItem", { events: { "workspaceSelected": "qx.event.type.Data", "workspaceUpdated": "qx.event.type.Data", - "deleteWorkspaceRequested": "qx.event.type.Data" + "trashWorkspaceRequested": "qx.event.type.Data", + "untrashWorkspaceRequested": "qx.event.type.Data", + "deleteWorkspaceRequested": "qx.event.type.Data", }, properties: { @@ -183,31 +185,46 @@ qx.Class.define("osparc.dashboard.WorkspaceButtonItem", { position: "bottom-right" }); - const editButton = new qx.ui.menu.Button(this.tr("Edit..."), "@FontAwesome5Solid/pencil-alt/12"); - editButton.addListener("execute", () => { - const workspace = this.getWorkspace(); - const workspaceEditor = new osparc.editor.WorkspaceEditor(workspace); - const title = this.tr("Edit Workspace"); - const win = osparc.ui.window.Window.popUpInWindow(workspaceEditor, title, 300, 150); - workspaceEditor.addListener("workspaceUpdated", () => { - win.close(); - this.fireDataEvent("workspaceUpdated", workspace.getWorkspaceId()); + const studyBrowserContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + if ( + studyBrowserContext === "search" || + studyBrowserContext === "workspaces" + ) { + const editButton = new qx.ui.menu.Button(this.tr("Edit..."), "@FontAwesome5Solid/pencil-alt/12"); + editButton.addListener("execute", () => { + const workspace = this.getWorkspace(); + const workspaceEditor = new osparc.editor.WorkspaceEditor(workspace); + const title = this.tr("Edit Workspace"); + const win = osparc.ui.window.Window.popUpInWindow(workspaceEditor, title, 300, 150); + workspaceEditor.addListener("workspaceUpdated", () => { + win.close(); + this.fireDataEvent("workspaceUpdated", workspace.getWorkspaceId()); + }); + workspaceEditor.addListener("cancel", () => win.close()); }); - workspaceEditor.addListener("cancel", () => win.close()); - }); - menu.add(editButton); + menu.add(editButton); + + const shareButton = new qx.ui.menu.Button(this.tr("Share..."), "@FontAwesome5Solid/share-alt/12"); + shareButton.addListener("execute", () => this.__openShareWith(), this); + menu.add(shareButton); - const shareButton = new qx.ui.menu.Button(this.tr("Share..."), "@FontAwesome5Solid/share-alt/12"); - shareButton.addListener("execute", () => this.__openShareWith(), this); - menu.add(shareButton); + menu.addSeparator(); - menu.addSeparator(); + const trashButton = new qx.ui.menu.Button(this.tr("Trash"), "@FontAwesome5Solid/trash/12"); + trashButton.addListener("execute", () => this.__trashWorkspaceRequested(), this); + menu.add(trashButton); + } else if (studyBrowserContext === "trash") { + const restoreButton = new qx.ui.menu.Button(this.tr("Restore"), "@MaterialIcons/restore_from_trash/16"); + restoreButton.addListener("execute", () => this.fireDataEvent("untrashWorkspaceRequested", this.getWorkspace()), this); + menu.add(restoreButton); - const deleteButton = new qx.ui.menu.Button(this.tr("Delete"), "@FontAwesome5Solid/trash/12"); - osparc.utils.Utils.setIdToWidget(deleteButton, "deleteWorkspaceMenuItem"); - deleteButton.addListener("execute", () => this.__deleteWorkspaceRequested(), this); - menu.add(deleteButton); + menu.addSeparator(); + const deleteButton = new qx.ui.menu.Button(this.tr("Delete"), "@FontAwesome5Solid/trash/12"); + osparc.utils.Utils.setIdToWidget(deleteButton, "deleteWorkspaceMenuItem"); + deleteButton.addListener("execute", () => this.__deleteWorkspaceRequested(), this); + menu.add(deleteButton); + } menuButton.setMenu(menu); } }, @@ -237,7 +254,9 @@ qx.Class.define("osparc.dashboard.WorkspaceButtonItem", { }, __itemSelected: function(newVal) { - if (newVal) { + const studyBrowserContext = osparc.store.Store.getInstance().getStudyBrowserContext(); + // do not allow selecting workspace + if (studyBrowserContext !== "trash" && newVal) { this.fireDataEvent("workspaceSelected", this.getWorkspaceId()); } this.setValue(false); @@ -250,6 +269,24 @@ qx.Class.define("osparc.dashboard.WorkspaceButtonItem", { permissionsView.addListener("updateAccessRights", () => this.__applyAccessRights(this.getWorkspace().getAccessRights()), this); }, + __trashWorkspaceRequested: function() { + const trashDays = osparc.store.StaticInfo.getInstance().getTrashRetentionDays(); + let msg = this.tr("Are you sure you want to move the Workspace and all its content to the trash?"); + msg += "

" + this.tr("It will be permanently deleted after ") + trashDays + " days."; + const confirmationWin = new osparc.ui.window.Confirmation(msg).set({ + caption: this.tr("Move to Trash"), + confirmText: this.tr("Move to Trash"), + confirmAction: "delete" + }); + confirmationWin.center(); + confirmationWin.open(); + confirmationWin.addListener("close", () => { + if (confirmationWin.getConfirmed()) { + this.fireDataEvent("trashWorkspaceRequested", this.getWorkspaceId()); + } + }, this); + }, + __deleteWorkspaceRequested: function() { let msg = this.tr("Are you sure you want to delete") + " " + this.getTitle() + "?"; msg += "
" + this.tr("All the content of the workspace will be deleted."); diff --git a/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTree.js b/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTree.js index 533c90c4643..da5a21797f1 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTree.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTree.js @@ -46,7 +46,7 @@ qx.Class.define("osparc.dashboard.WorkspacesAndFoldersTree", { this.__initTree(); // preselect "My Workspace" - this.contextChanged(null, null); + this.contextChanged("studiesAndFolders"); osparc.store.Folders.getInstance().addListener("folderAdded", e => { const folder = e.getData(); diff --git a/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTreeItem.js b/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTreeItem.js index 13793622cb7..75f120a86c5 100644 --- a/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTreeItem.js +++ b/services/static-webserver/client/source/class/osparc/dashboard/WorkspacesAndFoldersTreeItem.js @@ -24,6 +24,7 @@ qx.Class.define("osparc.dashboard.WorkspacesAndFoldersTreeItem", { this.set({ indent: 12, // defaults to 19, decorator: "rounded", + padding: 2, maxWidth: osparc.dashboard.ResourceBrowserBase.SIDE_SPACER_WIDTH - 12, }); diff --git a/services/static-webserver/client/source/class/osparc/data/Resources.js b/services/static-webserver/client/source/class/osparc/data/Resources.js index 8ff5eb822ba..ad1b51cc840 100644 --- a/services/static-webserver/client/source/class/osparc/data/Resources.js +++ b/services/static-webserver/client/source/class/osparc/data/Resources.js @@ -119,6 +119,16 @@ qx.Class.define("osparc.data.Resources", { method: "GET", url: statics.API + "/projects?type=user" }, + getOne: { + useCache: false, + method: "GET", + url: statics.API + "/projects/{studyId}" + }, + getActive: { + useCache: false, + method: "GET", + url: statics.API + "/projects/active?client_session_id={tabId}" + }, getPage: { useCache: false, method: "GET", @@ -129,15 +139,10 @@ qx.Class.define("osparc.data.Resources", { method: "GET", url: statics.API + "/projects:search?offset={offset}&limit={limit}&text={text}&tag_ids={tagIds}&order_by={orderBy}" }, - getOne: { - useCache: false, - method: "GET", - url: statics.API + "/projects/{studyId}" - }, - getActive: { + getPageTrashed: { useCache: false, method: "GET", - url: statics.API + "/projects/active?client_session_id={tabId}" + url: statics.API + "/projects?filters={%22trashed%22:%22true%22}&offset={offset}&limit={limit}&order_by={orderBy}" }, postToTemplate: { method: "POST", @@ -187,6 +192,14 @@ qx.Class.define("osparc.data.Resources", { method: "PATCH", url: statics.API + "/projects/{studyId}" }, + trash: { + method: "POST", + url: statics.API + "/projects/{studyId}:trash" + }, + untrash: { + method: "POST", + url: statics.API + "/projects/{studyId}:untrash" + }, delete: { method: "DELETE", url: statics.API + "/projects/{studyId}" @@ -301,14 +314,19 @@ qx.Class.define("osparc.data.Resources", { method: "GET", url: statics.API + "/folders?workspace_id={workspaceId}&folder_id={folderId}&offset={offset}&limit={limit}&order_by={orderBy}" }, + getOne: { + method: "GET", + url: statics.API + "/folders/{folderId}" + }, getPageSearch: { useCache: false, method: "GET", url: statics.API + "/folders:search?offset={offset}&limit={limit}&text={text}&order_by={orderBy}" }, - getOne: { + getPageTrashed: { + useCache: false, method: "GET", - url: statics.API + "/folders/{folderId}" + url: statics.API + "/folders?filters={%22trashed%22:%22true%22}&offset={offset}&limit={limit}&order_by={orderBy}" }, post: { method: "POST", @@ -326,6 +344,14 @@ qx.Class.define("osparc.data.Resources", { method: "PUT", url: statics.API + "/folders/{folderId}/folders/{workspaceId}" }, + trash: { + method: "POST", + url: statics.API + "/folders/{folderId}:trash" + }, + untrash: { + method: "POST", + url: statics.API + "/folders/{folderId}:untrash" + }, } }, "workspaces": { @@ -338,6 +364,16 @@ qx.Class.define("osparc.data.Resources", { method: "GET", url: statics.API + "/workspaces/{workspaceId}" }, + getPageSearch: { + useCache: false, + method: "GET", + url: statics.API + "/workspaces:search?offset={offset}&limit={limit}&text={text}&order_by={orderBy}" + }, + getPageTrashed: { + useCache: false, + method: "GET", + url: statics.API + "/workspaces?filters={%22trashed%22:%22true%22}&offset={offset}&limit={limit}&order_by={orderBy}" + }, post: { method: "POST", url: statics.API + "/workspaces" @@ -350,6 +386,14 @@ qx.Class.define("osparc.data.Resources", { method: "DELETE", url: statics.API + "/workspaces/{workspaceId}" }, + trash: { + method: "POST", + url: statics.API + "/workspaces/{workspaceId}:trash" + }, + untrash: { + method: "POST", + url: statics.API + "/workspaces/{workspaceId}:untrash" + }, postAccessRights: { method: "POST", url: statics.API + "/workspaces/{workspaceId}/groups/{groupId}" @@ -402,6 +446,18 @@ qx.Class.define("osparc.data.Resources", { } }, + /* + * TRASH + */ + "trash": { + endpoints: { + delete: { + method: "DELETE", + url: statics.API + "/trash" + } + } + }, + /* * SNAPSHOTS */ diff --git a/services/static-webserver/client/source/class/osparc/data/model/Folder.js b/services/static-webserver/client/source/class/osparc/data/model/Folder.js index b8b9eb03b21..6a255a45dc5 100644 --- a/services/static-webserver/client/source/class/osparc/data/model/Folder.js +++ b/services/static-webserver/client/source/class/osparc/data/model/Folder.js @@ -37,7 +37,7 @@ qx.Class.define("osparc.data.model.Folder", { owner: folderData.owner, createdAt: new Date(folderData.createdAt), lastModified: new Date(folderData.modifiedAt), - trashedAt: folderData.trashedAt ? new Date(folderData.trashedAt) : this.getTrashedAt(), + trashedAt: folderData.trashedAt ? new Date(folderData.trashedAt) : null, }); }, diff --git a/services/static-webserver/client/source/class/osparc/data/model/IframeHandler.js b/services/static-webserver/client/source/class/osparc/data/model/IframeHandler.js index fa037642af4..583b31f979e 100644 --- a/services/static-webserver/client/source/class/osparc/data/model/IframeHandler.js +++ b/services/static-webserver/client/source/class/osparc/data/model/IframeHandler.js @@ -300,7 +300,7 @@ qx.Class.define("osparc.data.model.IframeHandler", { if (response.status < 400) { this.__serviceReadyIn(srvUrl); } else { - console.log(`Connecting: ${srvUrl} is not reachable. Status: ${response.status}`); + console.error(`Connecting: ${srvUrl} is not reachable. Status: ${response.status}`); retry(); } }) diff --git a/services/static-webserver/client/source/class/osparc/data/model/Workspace.js b/services/static-webserver/client/source/class/osparc/data/model/Workspace.js index 56023d1eb4e..65592fb1789 100644 --- a/services/static-webserver/client/source/class/osparc/data/model/Workspace.js +++ b/services/static-webserver/client/source/class/osparc/data/model/Workspace.js @@ -37,6 +37,8 @@ qx.Class.define("osparc.data.model.Workspace", { accessRights: workspaceData.accessRights, createdAt: new Date(workspaceData.createdAt), modifiedAt: new Date(workspaceData.modifiedAt), + trashedAt: workspaceData.trashedAt ? new Date(workspaceData.trashedAt) : null, + trashedBy: workspaceData.trashedBy, }); }, @@ -95,7 +97,19 @@ qx.Class.define("osparc.data.model.Workspace", { nullable: true, init: null, event: "changeModifiedAt" - } + }, + + trashedAt: { + check: "Date", + nullable: true, + init: null, + }, + + trashedBy: { + check: "Number", + nullable: true, + init: null, + }, }, statics: { diff --git a/services/static-webserver/client/source/class/osparc/store/Folders.js b/services/static-webserver/client/source/class/osparc/store/Folders.js index d6e83d8fb23..4206cb212dd 100644 --- a/services/static-webserver/client/source/class/osparc/store/Folders.js +++ b/services/static-webserver/client/source/class/osparc/store/Folders.js @@ -78,6 +78,33 @@ qx.Class.define("osparc.store.Folders", { }); }, + fetchAllTrashedFolders: function(orderBy = { + field: "modified_at", + direction: "desc" + }) { + if (osparc.auth.Data.getInstance().isGuest()) { + return new Promise(resolve => { + resolve([]); + }); + } + + const curatedOrderBy = this.self().curateOrderBy(orderBy); + const params = { + url: { + orderBy: JSON.stringify(curatedOrderBy), + } + }; + return osparc.data.Resources.getInstance().getAllPages("folders", params, "getPageTrashed") + .then(trashedFoldersData => { + const folders = []; + trashedFoldersData.forEach(folderData => { + const folder = this.__addToCache(folderData); + folders.push(folder); + }); + return folders; + }); + }, + searchFolders: function( text, orderBy = { @@ -126,6 +153,37 @@ qx.Class.define("osparc.store.Folders", { }); }, + trashFolder: function(folderId, workspaceId) { + const params = { + "url": { + folderId + } + }; + return osparc.data.Resources.getInstance().fetch("folders", "trash", params) + .then(() => { + const folder = this.getFolder(folderId); + if (folder) { + this.__deleteFromCache(folderId, workspaceId); + this.fireDataEvent("folderRemoved", folder); + } + }) + .catch(console.error); + }, + + untrashFolder: function(folder) { + const params = { + "url": { + folderId: folder.getFolderId(), + } + }; + return osparc.data.Resources.getInstance().fetch("folders", "untrash", params) + .then(() => { + this.foldersCached.unshift(folder); + this.fireDataEvent("folderAdded", folder); + }) + .catch(console.error); + }, + deleteFolder: function(folderId, workspaceId) { const params = { "url": { diff --git a/services/static-webserver/client/source/class/osparc/store/StaticInfo.js b/services/static-webserver/client/source/class/osparc/store/StaticInfo.js index 1681a801cf4..2ac96fd58b0 100644 --- a/services/static-webserver/client/source/class/osparc/store/StaticInfo.js +++ b/services/static-webserver/client/source/class/osparc/store/StaticInfo.js @@ -69,6 +69,16 @@ qx.Class.define("osparc.store.StaticInfo", { return null; }, + getTrashRetentionDays: function() { + const staticKey = "webserverProjects"; + const wsStaticData = this.getValue(staticKey); + const key = "PROJECTS_TRASH_RETENTION_DAYS"; + if (key in wsStaticData) { + return wsStaticData[key]; + } + return "unknown"; + }, + getAccountDeletionRetentionDays: function() { const staticKey = "webserverLogin"; const wsStaticData = this.getValue(staticKey); diff --git a/services/static-webserver/client/source/class/osparc/store/Store.js b/services/static-webserver/client/source/class/osparc/store/Store.js index d2c2d24104e..7b94b336852 100644 --- a/services/static-webserver/client/source/class/osparc/store/Store.js +++ b/services/static-webserver/client/source/class/osparc/store/Store.js @@ -67,7 +67,7 @@ qx.Class.define("osparc.store.Store", { nullable: true }, studyBrowserContext: { - check: ["studiesAndFolders", "workspaces", "search"], + check: ["studiesAndFolders", "workspaces", "search", "trash"], init: "studiesAndFolders", nullable: false, event: "changeStudyBrowserContext", @@ -413,6 +413,43 @@ qx.Class.define("osparc.store.Store", { } }, + trashStudy: function(studyId) { + const params = { + url: { + studyId + } + }; + return new Promise((resolve, reject) => { + osparc.data.Resources.fetch("studies", "trash", params) + .then(() => { + this.remove("studies", "uuid", studyId); + resolve(); + }) + .catch(err => { + console.error(err); + reject(err); + }); + }); + }, + + untrashStudy: function(studyId) { + const params = { + url: { + studyId + } + }; + return new Promise((resolve, reject) => { + osparc.data.Resources.fetch("studies", "untrash", params) + .then(() => { + resolve(); + }) + .catch(err => { + console.error(err); + reject(err); + }); + }); + }, + getTemplate: function(templateId) { const templates = this.getTemplates(); return templates.find(template => template["uuid"] === templateId); diff --git a/services/static-webserver/client/source/class/osparc/store/Workspaces.js b/services/static-webserver/client/source/class/osparc/store/Workspaces.js index 253ac714a1d..924312639de 100644 --- a/services/static-webserver/client/source/class/osparc/store/Workspaces.js +++ b/services/static-webserver/client/source/class/osparc/store/Workspaces.js @@ -46,6 +46,15 @@ qx.Class.define("osparc.store.Workspaces", { thumbnail, }; }, + + curateOrderBy: function(orderBy) { + const curatedOrderBy = osparc.utils.Utils.deepCloneObject(orderBy); + if (curatedOrderBy.field !== "name") { + // only "modified_at" and "name" supported + curatedOrderBy.field = "modified_at"; + } + return curatedOrderBy; + }, }, members: { @@ -61,26 +70,105 @@ qx.Class.define("osparc.store.Workspaces", { return osparc.data.Resources.getInstance().getAllPages("workspaces") .then(workspacesData => { workspacesData.forEach(workspaceData => { - const workspace = new osparc.data.model.Workspace(workspaceData); - this.__addToCache(workspace); + this.__addToCache(workspaceData); }); return this.workspacesCached; }); }, + fetchAllTrashedWorkspaces: function(orderBy = { + field: "modified_at", + direction: "desc" + }) { + if (osparc.auth.Data.getInstance().isGuest()) { + return new Promise(resolve => { + resolve([]); + }); + } + + const curatedOrderBy = this.self().curateOrderBy(orderBy); + const params = { + url: { + orderBy: JSON.stringify(curatedOrderBy), + } + }; + return osparc.data.Resources.getInstance().getAllPages("workspaces", params, "getPageTrashed") + .then(trashedWorkspacesData => { + const workspaces = []; + trashedWorkspacesData.forEach(workspaceData => { + const workspace = this.__addToCache(workspaceData); + workspaces.push(workspace); + }); + return workspaces; + }); + }, + + searchWorkspaces: function(text) { + if (osparc.auth.Data.getInstance().isGuest()) { + return new Promise(resolve => { + resolve([]); + }); + } + + const params = { + url: { + text, + } + }; + return osparc.data.Resources.getInstance().getAllPages("workspaces", params, "getPageSearch") + .then(workspacesData => { + const workspaces = []; + workspacesData.forEach(workspaceData => { + const workspace = this.__addToCache(workspaceData); + workspaces.push(workspace); + }); + return workspaces; + }); + }, + postWorkspace: function(newWorkspaceData) { const params = { data: newWorkspaceData }; return osparc.data.Resources.getInstance().fetch("workspaces", "post", params) .then(workspaceData => { - const newWorkspace = new osparc.data.model.Workspace(workspaceData); - this.__addToCache(newWorkspace); + const newWorkspace = this.__addToCache(workspaceData); this.fireDataEvent("workspaceAdded", newWorkspace); return newWorkspace; }); }, + trashWorkspace: function(workspaceId) { + const params = { + "url": { + workspaceId + } + }; + return osparc.data.Resources.getInstance().fetch("workspaces", "trash", params) + .then(() => { + const workspace = this.getWorkspace(workspaceId); + if (workspace) { + this.__deleteFromCache(workspaceId); + this.fireDataEvent("workspaceRemoved", workspace); + } + }) + .catch(console.error); + }, + + untrashWorkspace: function(workspace) { + const params = { + "url": { + workspaceId: workspace.getWorkspaceId(), + } + }; + return osparc.data.Resources.getInstance().fetch("workspaces", "untrash", params) + .then(() => { + this.workspacesCached.unshift(workspace); + this.fireDataEvent("workspaceAdded", workspace); + }) + .catch(console.error); + }, + deleteWorkspace: function(workspaceId) { const params = { "url": { @@ -201,11 +289,27 @@ qx.Class.define("osparc.store.Workspaces", { return this.workspacesCached; }, - __addToCache: function(workspace) { - const found = this.workspacesCached.find(w => w.getWorkspaceId() === workspace.getWorkspaceId()); - if (!found) { + __addToCache: function(workspaceData) { + let workspace = this.workspacesCached.find(w => w.getWorkspaceId() === workspaceData["workspaceId"]); + if (workspace) { + const props = Object.keys(qx.util.PropertyUtil.getProperties(osparc.data.model.Workspace)); + // put + Object.keys(workspaceData).forEach(key => { + if (key === "createdAt") { + workspace.set("createdAt", new Date(workspaceData["createdAt"])); + } else if (key === "modifiedAt") { + workspace.set("modifiedAt", new Date(workspaceData["modifiedAt"])); + } else if (key === "trashedAt") { + workspace.set("trashedAt", new Date(workspaceData["trashedAt"])); + } else if (props.includes(key)) { + workspace.set(key, workspaceData[key]); + } + }); + } else { + workspace = new osparc.data.model.Workspace(workspaceData); this.workspacesCached.unshift(workspace); } + return workspace; }, __deleteFromCache: function(workspaceId) {