From 22baefca224769456eb6de146ef444c27b7f774e Mon Sep 17 00:00:00 2001 From: Trae Yelovich Date: Wed, 27 Sep 2023 09:40:02 -0400 Subject: [PATCH 1/6] fix(job,uss): Remove extra 'delete job' msg; Add check for paste API & fix paste refresh Signed-off-by: Trae Yelovich --- .../i18n/sample/src/uss/actions.i18n.json | 3 +- packages/zowe-explorer/src/job/actions.ts | 4 +-- packages/zowe-explorer/src/uss/ZoweUSSNode.ts | 31 +++++++------------ packages/zowe-explorer/src/uss/actions.ts | 13 +++----- 4 files changed, 19 insertions(+), 32 deletions(-) diff --git a/packages/zowe-explorer/i18n/sample/src/uss/actions.i18n.json b/packages/zowe-explorer/i18n/sample/src/uss/actions.i18n.json index a4af6ec483..638518fe52 100644 --- a/packages/zowe-explorer/i18n/sample/src/uss/actions.i18n.json +++ b/packages/zowe-explorer/i18n/sample/src/uss/actions.i18n.json @@ -12,5 +12,6 @@ "deleteUssPrompt.confirmation.message": "Are you sure you want to delete the following item?\nThis will permanently remove the following file or folder from your system.\n\n{0}", "deleteUssPrompt.confirmation.cancel.log.debug": "Delete action was canceled.", "ZoweUssNode.copyDownload.progress": "Copying file structure...", - "ZoweUssNode.copyUpload.progress": "Pasting files..." + "ZoweUssNode.copyUpload.progress": "Pasting files...", + "uss.paste.apiNotAvailable": "The paste operation is not supported for this node." } diff --git a/packages/zowe-explorer/src/job/actions.ts b/packages/zowe-explorer/src/job/actions.ts index b7dd81ef9e..afe755dfdb 100644 --- a/packages/zowe-explorer/src/job/actions.ts +++ b/packages/zowe-explorer/src/job/actions.ts @@ -383,12 +383,10 @@ async function deleteSingleJob(job: IZoweJobTreeNode, jobsProvider: IZoweTree, jobsProvider: IZoweTree): Promise { diff --git a/packages/zowe-explorer/src/uss/ZoweUSSNode.ts b/packages/zowe-explorer/src/uss/ZoweUSSNode.ts index db0f1db14d..d45252e6a7 100644 --- a/packages/zowe-explorer/src/uss/ZoweUSSNode.ts +++ b/packages/zowe-explorer/src/uss/ZoweUSSNode.ts @@ -189,11 +189,16 @@ export class ZoweUSSNode extends ZoweTreeNode implements IZoweUSSTreeNode { return this.children; } + // If search path has changed, invalidate all children + if (this.fullPath?.length > 0 && this.prevPath !== this.fullPath) { + this.children = []; + } + // Build a list of nodes based on the API response const responseNodes: IZoweUSSTreeNode[] = []; - const newNodeCreated: boolean = response.apiResponse.items.reduce((lastResult: boolean, item) => { + for (const item of response.apiResponse.items) { if (item.name === "." || item.name === "..") { - return lastResult || false; + continue; } const existing = this.children.find( @@ -213,7 +218,7 @@ export class ZoweUSSNode extends ZoweTreeNode implements IZoweUSSTreeNode { }; responseNodes.push(existing); existing.onUpdateEmitter.fire(existing); - return lastResult || false; + continue; } if (item.mode.startsWith("d")) { @@ -253,20 +258,6 @@ export class ZoweUSSNode extends ZoweTreeNode implements IZoweUSSTreeNode { }; responseNodes.push(temp); } - - return lastResult || true; - }, false); - - this.dirty = false; - - // If no new nodes were created, return the cached list of children - if (!newNodeCreated) { - return this.children; - } - - // If search path has changed, invalidate all children - if (this.fullPath?.length > 0 && this.prevPath !== this.fullPath) { - this.children = []; } const nodesToAdd = responseNodes.filter((c) => !this.children.includes(c)); @@ -277,6 +268,7 @@ export class ZoweUSSNode extends ZoweTreeNode implements IZoweUSSTreeNode { .filter((c) => !nodesToRemove.includes(c)) .sort((a, b) => ((a.label as string) < (b.label as string) ? -1 : 1)); this.prevPath = this.fullPath; + this.dirty = false; return this.children; } @@ -749,10 +741,9 @@ export class ZoweUSSNode extends ZoweTreeNode implements IZoweUSSTreeNode { } const prof = this.getProfile(); - const remotePath = this.fullPath; try { + const fileTreeToPaste: UssFileTree = JSON.parse(clipboardContents); const api = ZoweExplorerApiRegister.getUssApi(this.profile); - const fileTreeToPaste: UssFileTree = JSON.parse(await vscode.env.clipboard.readText()); const sessionName = this.getSessionNode().getLabel() as string; const task: imperative.ITaskWithStatus = { @@ -767,7 +758,7 @@ export class ZoweUSSNode extends ZoweTreeNode implements IZoweUSSTreeNode { }; for (const subnode of fileTreeToPaste.children) { - await this.paste(sessionName, remotePath, { api, tree: subnode, options }); + await this.paste(sessionName, this.fullPath, { api, tree: subnode, options }); } } catch (error) { await errorHandling(error, this.label.toString(), localize("copyUssFile.error", "Error uploading files")); diff --git a/packages/zowe-explorer/src/uss/actions.ts b/packages/zowe-explorer/src/uss/actions.ts index c1ac4833f3..8ff901468d 100644 --- a/packages/zowe-explorer/src/uss/actions.ts +++ b/packages/zowe-explorer/src/uss/actions.ts @@ -487,21 +487,18 @@ export async function pasteUssFile(ussFileProvider: IZoweTree, */ export async function pasteUss(ussFileProvider: IZoweTree, node: IZoweUSSTreeNode): Promise { ZoweLogger.trace("uss.actions.pasteUss called."); - const a = ussFileProvider.getTreeView().selection as IZoweUSSTreeNode[]; - let selectedNode = node; - if (!selectedNode) { - selectedNode = a.length > 0 ? a[0] : (a as unknown as IZoweUSSTreeNode); + if (node.pasteUssTree == null && node.copyUssFile == null) { + await Gui.infoMessage(localize("uss.paste.apiNotAvailable", "The paste operation is not supported for this node.")); + return; } - await Gui.withProgress( { location: vscode.ProgressLocation.Window, title: localize("ZoweUssNode.copyUpload.progress", "Pasting files..."), }, async () => { - await (selectedNode.pasteUssTree ? selectedNode.pasteUssTree() : selectedNode.copyUssFile()); + await (node.pasteUssTree ? node.pasteUssTree() : node.copyUssFile()); } ); - const nodeToRefresh = node?.contextValue != null && contextually.isUssSession(node) ? selectedNode : selectedNode.getParent(); - ussFileProvider.refreshElement(nodeToRefresh); + ussFileProvider.refreshElement(node); } From 1f36769207367e1d6e64e230d384b81112bf7cd7 Mon Sep 17 00:00:00 2001 From: Trae Yelovich Date: Wed, 27 Sep 2023 09:48:50 -0400 Subject: [PATCH 2/6] chore: update CHANGELOG Signed-off-by: Trae Yelovich --- packages/zowe-explorer/CHANGELOG.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/packages/zowe-explorer/CHANGELOG.md b/packages/zowe-explorer/CHANGELOG.md index d6550315ec..0001bbd025 100644 --- a/packages/zowe-explorer/CHANGELOG.md +++ b/packages/zowe-explorer/CHANGELOG.md @@ -11,6 +11,9 @@ All notable changes to the "vscode-extension-for-zowe" extension will be documen ### Bug fixes - Fixed submitting local JCL using command pallet option `Zowe Explorer: Submit JCL` by adding a check for chosen profile returned to continue the action. [#1625](https://github.com/zowe/vscode-extension-for-zowe/issues/1625) +- Fixed issue where USS nodes were not removed from tree during deletion. [#2479](https://github.com/zowe/vscode-extension-for-zowe/issues/2479) +- Fixed issue where new USS nodes from a paste operation were not shown in tree until refreshed. [#2479](https://github.com/zowe/vscode-extension-for-zowe/issues/2479) +- Fixed issue where the "Delete Job" action showed a successful deletion message, even if the API returned an error. ## `2.11.0` From 9d74b9b18b3c2ec956c36b18c6d1a3d0136d6885 Mon Sep 17 00:00:00 2001 From: Trae Yelovich Date: Wed, 27 Sep 2023 10:04:42 -0400 Subject: [PATCH 3/6] test(uss): Add test case for pasteUss - unsupported APIs Signed-off-by: Trae Yelovich --- .../__tests__/__unit__/uss/actions.unit.test.ts | 14 +++++++++++--- 1 file changed, 11 insertions(+), 3 deletions(-) diff --git a/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts index 4c84b3d6ec..797f1ee46b 100644 --- a/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts @@ -891,9 +891,7 @@ describe("USS Action Unit Tests - copy file / directory", () => { it("tests pasteUssFile executed successfully with selected nodes", async () => { const globalMocks = createGlobalMocks(); const blockMocks = await createBlockMocks(globalMocks); - const parent = blockMocks.treeNodes.testUSSTree.getTreeView(); - parent.selection = blockMocks.nodes[0]; - await ussNodeActions.pasteUssFile(blockMocks.treeNodes.testUSSTree, undefined); + await ussNodeActions.pasteUssFile(blockMocks.treeNodes.testUSSTree, blockMocks.nodes[0]); expect(sharedUtils.getSelectedNodeList(blockMocks.treeNodes.ussNode, blockMocks.treeNodes.ussNodes)).toEqual([blockMocks.treeNodes.ussNode]); }); it("tests pasteUssFile executed successfully with one node", async () => { @@ -905,6 +903,16 @@ describe("USS Action Unit Tests - copy file / directory", () => { await ussNodeActions.pasteUssFile(blockMocks.treeNodes.testUSSTree, blockMocks.nodes[0]); expect(sharedUtils.getSelectedNodeList(blockMocks.treeNodes.ussNode, blockMocks.treeNodes.ussNodes)).toEqual([blockMocks.treeNodes.ussNode]); }); + it("tests pasteUss returns early if APIs are not supported", async () => { + const globalMocks = createGlobalMocks(); + const blockMocks = await createBlockMocks(globalMocks); + const testNode = blockMocks.nodes[0]; + testNode.copyUssFile = testNode.pasteUssTree = null; + const infoMessageSpy = jest.spyOn(Gui, "infoMessage"); + await ussNodeActions.pasteUss(blockMocks.treeNodes.testUSSTree, testNode); + expect(infoMessageSpy).toHaveBeenCalledWith("The paste operation is not supported for this node."); + infoMessageSpy.mockRestore(); + }); }); describe("USS Action Unit Tests - function deleteUSSFilesPrompt", () => { From f5a19bf772368b0d98ae58c2fc17e7e9f43e766e Mon Sep 17 00:00:00 2001 From: Trae Yelovich Date: Wed, 27 Sep 2023 10:45:16 -0400 Subject: [PATCH 4/6] fix(uss): update USS directory icons when collapsed Signed-off-by: Trae Yelovich --- packages/zowe-explorer/CHANGELOG.md | 1 + packages/zowe-explorer/src/uss/USSTree.ts | 9 +++++++++ 2 files changed, 10 insertions(+) diff --git a/packages/zowe-explorer/CHANGELOG.md b/packages/zowe-explorer/CHANGELOG.md index 0001bbd025..e73a25663c 100644 --- a/packages/zowe-explorer/CHANGELOG.md +++ b/packages/zowe-explorer/CHANGELOG.md @@ -14,6 +14,7 @@ All notable changes to the "vscode-extension-for-zowe" extension will be documen - Fixed issue where USS nodes were not removed from tree during deletion. [#2479](https://github.com/zowe/vscode-extension-for-zowe/issues/2479) - Fixed issue where new USS nodes from a paste operation were not shown in tree until refreshed. [#2479](https://github.com/zowe/vscode-extension-for-zowe/issues/2479) - Fixed issue where the "Delete Job" action showed a successful deletion message, even if the API returned an error. +- USS directories and sessions now update with their respective "collapsed icon" when collapsed. ## `2.11.0` diff --git a/packages/zowe-explorer/src/uss/USSTree.ts b/packages/zowe-explorer/src/uss/USSTree.ts index c53a3b3fd3..ed2d116179 100644 --- a/packages/zowe-explorer/src/uss/USSTree.ts +++ b/packages/zowe-explorer/src/uss/USSTree.ts @@ -80,6 +80,15 @@ export class USSTree extends ZoweTreeProvider implements IZoweTree { + if (contextually.isUssDirectory(e.element)) { + const newIcon = getIconByNode(e.element); + if (newIcon) { + e.element.iconPath = newIcon; + this.mOnDidChangeTreeData.fire(e.element); + } + } + }) } /** From 51fb511b2d5f065b8f1c28a09f22d22059cd6947 Mon Sep 17 00:00:00 2001 From: Trae Yelovich Date: Wed, 27 Sep 2023 11:50:18 -0400 Subject: [PATCH 5/6] fix(tests): Add missing 'onDidCollapseElement' function to mocks Signed-off-by: Trae Yelovich --- .../__tests__/__unit__/globals/Gui.unit.test.ts | 2 +- .../__tests__/__unit__/Profiles.extended.unit.test.ts | 5 ++++- .../__tests__/__unit__/ZoweExplorerExtender.unit.test.ts | 5 ++++- .../__tests__/__unit__/abstract/TreeProvider.unit.test.ts | 2 +- .../__tests__/__unit__/abstract/ZoweSaveQueue.unit.test.ts | 1 + .../__tests__/__unit__/dataset/DatasetTree.unit.test.ts | 5 ++++- .../zowe-explorer/__tests__/__unit__/extension.unit.test.ts | 2 +- .../__tests__/__unit__/generators/icons.unit.test.ts | 2 +- .../__tests__/__unit__/generators/messages.unit.test.ts | 2 +- .../__tests__/__unit__/job/ZosJobsProvider.unit.test.ts | 5 ++--- .../__tests__/__unit__/job/ZoweJobNode.unit.test.ts | 1 + .../__tests__/__unit__/job/actions.unit.test.ts | 1 + .../__tests__/__unit__/shared/actions.unit.test.ts | 5 ++++- .../__tests__/__unit__/shared/refresh.unit.test.ts | 2 +- .../__tests__/__unit__/uss/USSTree.unit.test.ts | 2 +- .../__tests__/__unit__/uss/ZoweUSSNode.unit.test.ts | 5 ++++- .../__tests__/__unit__/uss/actions.unit.test.ts | 2 +- .../__tests__/__unit__/utils/SessionUtils.unit.test.ts | 5 ++++- 18 files changed, 37 insertions(+), 17 deletions(-) diff --git a/packages/zowe-explorer-api/__tests__/__unit__/globals/Gui.unit.test.ts b/packages/zowe-explorer-api/__tests__/__unit__/globals/Gui.unit.test.ts index 28a11d19cc..a8382a571a 100644 --- a/packages/zowe-explorer-api/__tests__/__unit__/globals/Gui.unit.test.ts +++ b/packages/zowe-explorer-api/__tests__/__unit__/globals/Gui.unit.test.ts @@ -22,7 +22,7 @@ function createGlobalMocks() { showWarningMessage: jest.fn(), createOutputChannel: jest.fn(), createQuickPick: jest.fn(), - createTreeView: jest.fn(), + createTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), createWebviewPanel: jest.fn(), withProgress: jest.fn(), showTextDocument: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/Profiles.extended.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/Profiles.extended.unit.test.ts index 2876032d7b..5309decf5a 100644 --- a/packages/zowe-explorer/__tests__/__unit__/Profiles.extended.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/Profiles.extended.unit.test.ts @@ -112,7 +112,10 @@ async function createGlobalMocks() { configurable: true, }); Object.defineProperty(globals, "ISTHEIA", { get: () => false, configurable: true }); - Object.defineProperty(vscode.window, "createTreeView", { value: jest.fn(), configurable: true }); + Object.defineProperty(vscode.window, "createTreeView", { + value: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), + configurable: true, + }); Object.defineProperty(vscode.workspace, "getConfiguration", { value: newMocks.mockGetConfiguration, configurable: true, diff --git a/packages/zowe-explorer/__tests__/__unit__/ZoweExplorerExtender.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/ZoweExplorerExtender.unit.test.ts index 6ae337f750..34cbb87bb6 100644 --- a/packages/zowe-explorer/__tests__/__unit__/ZoweExplorerExtender.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/ZoweExplorerExtender.unit.test.ts @@ -53,7 +53,10 @@ describe("ZoweExplorerExtender unit tests", () => { }) .mockReturnValue(newMocks.profiles), }); - Object.defineProperty(vscode.window, "createTreeView", { value: jest.fn(), configurable: true }); + Object.defineProperty(vscode.window, "createTreeView", { + value: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), + configurable: true, + }); Object.defineProperty(vscode.window, "showErrorMessage", { value: newMocks.mockErrorMessage, configurable: true, diff --git a/packages/zowe-explorer/__tests__/__unit__/abstract/TreeProvider.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/abstract/TreeProvider.unit.test.ts index 096d54cf5b..0771a57f74 100644 --- a/packages/zowe-explorer/__tests__/__unit__/abstract/TreeProvider.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/abstract/TreeProvider.unit.test.ts @@ -37,7 +37,7 @@ async function createGlobalMocks() { mockLoadNamedProfile: jest.fn(), mockDefaultProfile: jest.fn(), withProgress: jest.fn(), - createTreeView: jest.fn(), + createTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), mockAffects: jest.fn(), mockEditSession: jest.fn(), mockCheckCurrentProfile: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/abstract/ZoweSaveQueue.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/abstract/ZoweSaveQueue.unit.test.ts index 048fb60b96..d50db952a2 100644 --- a/packages/zowe-explorer/__tests__/__unit__/abstract/ZoweSaveQueue.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/abstract/ZoweSaveQueue.unit.test.ts @@ -20,6 +20,7 @@ import { ZoweLogger } from "../../../src/utils/LoggerUtils"; describe("ZoweSaveQueue - unit tests", () => { const createGlobalMocks = () => { + jest.spyOn(Gui, "createTreeView").mockReturnValue({ onDidCollapseElement: jest.fn() } as any); const globalMocks = { errorMessageSpy: jest.spyOn(Gui, "errorMessage"), markDocumentUnsavedSpy: jest.spyOn(workspaceUtils, "markDocumentUnsaved"), diff --git a/packages/zowe-explorer/__tests__/__unit__/dataset/DatasetTree.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/dataset/DatasetTree.unit.test.ts index 9cb5357619..42d3a43fc0 100644 --- a/packages/zowe-explorer/__tests__/__unit__/dataset/DatasetTree.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/dataset/DatasetTree.unit.test.ts @@ -59,7 +59,10 @@ function createGlobalMocks() { globalMocks.mockProfileInstance = createInstanceOfProfile(globalMocks.testProfileLoaded); - Object.defineProperty(vscode.window, "createTreeView", { value: jest.fn(), configurable: true }); + Object.defineProperty(vscode.window, "createTreeView", { + value: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), + configurable: true, + }); Object.defineProperty(Gui, "showMessage", { value: jest.fn(), configurable: true }); Object.defineProperty(Gui, "setStatusBarMessage", { value: jest.fn().mockReturnValue({ dispose: jest.fn() }), configurable: true }); Object.defineProperty(vscode.window, "showTextDocument", { value: jest.fn(), configurable: true }); diff --git a/packages/zowe-explorer/__tests__/__unit__/extension.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/extension.unit.test.ts index 0f637993a6..30f3380063 100644 --- a/packages/zowe-explorer/__tests__/__unit__/extension.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/extension.unit.test.ts @@ -45,7 +45,7 @@ async function createGlobalMocks() { mockMoveSync: jest.fn(), mockGetAllProfileNames: jest.fn(), mockReveal: jest.fn(), - mockCreateTreeView: jest.fn(), + mockCreateTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), mockExecuteCommand: jest.fn(), mockRegisterCommand: jest.fn(), mockOnDidSaveTextDocument: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/generators/icons.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/generators/icons.unit.test.ts index 957c70970c..c9ec8046f2 100644 --- a/packages/zowe-explorer/__tests__/__unit__/generators/icons.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/generators/icons.unit.test.ts @@ -19,7 +19,7 @@ import * as vscode from "vscode"; describe("Checking icon generator's basics", () => { const setGlobalMocks = () => { - const createTreeView = jest.fn(); + const createTreeView = jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }); const getConfiguration = jest.fn(); Object.defineProperty(vscode.window, "createTreeView", { value: createTreeView }); diff --git a/packages/zowe-explorer/__tests__/__unit__/generators/messages.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/generators/messages.unit.test.ts index 2b46ac7d6a..a4fd28edc8 100644 --- a/packages/zowe-explorer/__tests__/__unit__/generators/messages.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/generators/messages.unit.test.ts @@ -19,7 +19,7 @@ jest.mock("vscode"); describe("Checking message generator's basics", () => { const setGlobalMocks = () => { - const createTreeView = jest.fn(); + const createTreeView = jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }); const getConfiguration = jest.fn(); Object.defineProperty(vscode.window, "createTreeView", { value: createTreeView }); diff --git a/packages/zowe-explorer/__tests__/__unit__/job/ZosJobsProvider.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/job/ZosJobsProvider.unit.test.ts index da5b74ffa7..7965f68cef 100644 --- a/packages/zowe-explorer/__tests__/__unit__/job/ZosJobsProvider.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/job/ZosJobsProvider.unit.test.ts @@ -45,7 +45,7 @@ async function createGlobalMocks() { mockGetJob: jest.fn(), mockRefresh: jest.fn(), mockAffectsConfig: jest.fn(), - createTreeView: jest.fn(), + createTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), mockGetSpoolFiles: jest.fn(), mockDeleteJobs: jest.fn(), mockShowInputBox: jest.fn(), @@ -85,7 +85,7 @@ async function createGlobalMocks() { }; }), }; - + jest.spyOn(Gui, "createTreeView").mockImplementation(globalMocks.createTreeView); Object.defineProperty(ProfilesCache, "getConfigInstance", { value: jest.fn(() => { return { @@ -173,7 +173,6 @@ async function createGlobalMocks() { Object.defineProperty(ZoweLogger, "warn", { value: jest.fn(), configurable: true }); Object.defineProperty(ZoweLogger, "info", { value: jest.fn(), configurable: true }); Object.defineProperty(ZoweLogger, "trace", { value: jest.fn(), configurable: true }); - globalMocks.createTreeView.mockReturnValue("testTreeView"); globalMocks.testSessionNode = createJobSessionNode(globalMocks.testSession, globalMocks.testProfile); globalMocks.mockGetJob.mockReturnValue(globalMocks.testIJob); globalMocks.mockGetJobsByOwnerAndPrefix.mockReturnValue([globalMocks.testIJob, globalMocks.testIJobComplete]); diff --git a/packages/zowe-explorer/__tests__/__unit__/job/ZoweJobNode.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/job/ZoweJobNode.unit.test.ts index 2ab7070292..554aaf2670 100644 --- a/packages/zowe-explorer/__tests__/__unit__/job/ZoweJobNode.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/job/ZoweJobNode.unit.test.ts @@ -40,6 +40,7 @@ async function createGlobalMocks() { mockAffectsConfig: jest.fn(), createTreeView: jest.fn(() => ({ reveal: jest.fn(), + onDidCollapseElement: jest.fn(), })), mockCreateSessCfgFromArgs: jest.fn(), mockGetSpoolFiles: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/job/actions.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/job/actions.unit.test.ts index 85737a5c89..58b8a65a80 100644 --- a/packages/zowe-explorer/__tests__/__unit__/job/actions.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/job/actions.unit.test.ts @@ -46,6 +46,7 @@ import { ZosJobsProvider } from "../../../src/job/ZosJobsProvider"; const activeTextEditorDocument = jest.fn(); function createGlobalMocks() { + jest.spyOn(Gui, "createTreeView").mockReturnValue({ onDidCollapseElement: jest.fn() } as any); Object.defineProperty(vscode.workspace, "getConfiguration", { value: jest.fn().mockImplementation(() => new Map([["zowe.jobs.confirmSubmission", false]])), configurable: true, diff --git a/packages/zowe-explorer/__tests__/__unit__/shared/actions.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/shared/actions.unit.test.ts index 51e6c042a8..aec9ead9cf 100644 --- a/packages/zowe-explorer/__tests__/__unit__/shared/actions.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/shared/actions.unit.test.ts @@ -69,7 +69,10 @@ async function createGlobalMocks() { }), configurable: true, }); - Object.defineProperty(vscode.window, "createTreeView", { value: jest.fn(), configurable: true }); + Object.defineProperty(vscode.window, "createTreeView", { + value: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), + configurable: true, + }); Object.defineProperty(vscode.workspace, "getConfiguration", { value: jest.fn(), configurable: true }); Object.defineProperty(vscode.window, "showInformationMessage", { value: jest.fn(), configurable: true }); Object.defineProperty(vscode.window, "showInputBox", { value: jest.fn(), configurable: true }); diff --git a/packages/zowe-explorer/__tests__/__unit__/shared/refresh.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/shared/refresh.unit.test.ts index f306801e88..7dc0661da9 100644 --- a/packages/zowe-explorer/__tests__/__unit__/shared/refresh.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/shared/refresh.unit.test.ts @@ -31,7 +31,7 @@ import { ZoweLogger } from "../../../src/utils/LoggerUtils"; function createGlobalMocks() { const globalMocks = { session: createISessionWithoutCredentials(), - createTreeView: jest.fn(), + createTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), mockLog: jest.fn(), mockDebug: jest.fn(), mockError: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/uss/USSTree.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/uss/USSTree.unit.test.ts index 56d20ad5b0..8de2e871ab 100644 --- a/packages/zowe-explorer/__tests__/__unit__/uss/USSTree.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/uss/USSTree.unit.test.ts @@ -47,7 +47,7 @@ async function createGlobalMocks() { showInputBox: jest.fn(), filters: jest.fn(), getFilters: jest.fn(), - createTreeView: jest.fn(), + createTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), createQuickPick: jest.fn(), getConfiguration: jest.fn(), ZosmfSession: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/uss/ZoweUSSNode.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/uss/ZoweUSSNode.unit.test.ts index b80a6805f3..1a02dc1ea2 100644 --- a/packages/zowe-explorer/__tests__/__unit__/uss/ZoweUSSNode.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/uss/ZoweUSSNode.unit.test.ts @@ -137,7 +137,10 @@ async function createGlobalMocks() { configurable: true, }); Object.defineProperty(vscode.window, "showInputBox", { value: globalMocks.showInputBox, configurable: true }); - Object.defineProperty(vscode.window, "createTreeView", { value: jest.fn(), configurable: true }); + Object.defineProperty(vscode.window, "createTreeView", { + value: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), + configurable: true, + }); Object.defineProperty(zowe, "ZosmfSession", { value: globalMocks.ZosmfSession, configurable: true }); Object.defineProperty(globalMocks.ZosmfSession, "createSessCfgFromArgs", { value: globalMocks.createSessCfgFromArgs, diff --git a/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts index 797f1ee46b..4d054e113a 100644 --- a/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/uss/actions.unit.test.ts @@ -60,7 +60,7 @@ function createGlobalMocks() { setStatusBarMessage: jest.fn().mockReturnValue({ dispose: jest.fn() }), showWarningMessage: jest.fn(), showErrorMessage: jest.fn(), - createTreeView: jest.fn(), + createTreeView: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), fileToUSSFile: jest.fn(), Upload: jest.fn(), isBinaryFileSync: jest.fn(), diff --git a/packages/zowe-explorer/__tests__/__unit__/utils/SessionUtils.unit.test.ts b/packages/zowe-explorer/__tests__/__unit__/utils/SessionUtils.unit.test.ts index c97166e6de..4925d06b58 100644 --- a/packages/zowe-explorer/__tests__/__unit__/utils/SessionUtils.unit.test.ts +++ b/packages/zowe-explorer/__tests__/__unit__/utils/SessionUtils.unit.test.ts @@ -29,7 +29,10 @@ describe("SessionUtils removeSession Unit Tests", () => { newMocks.datasetSessionNode = createDatasetSessionNode(newMocks.session, newMocks.imperativeProfile); newMocks.testDatasetTree = createDatasetTree(newMocks.datasetSessionNode, newMocks.treeView); newMocks.testDatasetTree.addFileHistory("[profile1]: TEST.NODE"); - Object.defineProperty(vscode.window, "createTreeView", { value: jest.fn(), configurable: true }); + Object.defineProperty(vscode.window, "createTreeView", { + value: jest.fn().mockReturnValue({ onDidCollapseElement: jest.fn() }), + configurable: true, + }); Object.defineProperty(vscode, "ConfigurationTarget", { value: jest.fn(), configurable: true }); newMocks.mockGetConfiguration.mockReturnValue(createPersistentConfig()); Object.defineProperty(vscode.workspace, "getConfiguration", { From 6924a8ee0e1999af99a3e15748e0ace011f1bd32 Mon Sep 17 00:00:00 2001 From: Trae Yelovich Date: Wed, 27 Sep 2023 11:57:08 -0400 Subject: [PATCH 6/6] fix(trees): Update icon when folder/session is collapsed Signed-off-by: Trae Yelovich --- packages/zowe-explorer/CHANGELOG.md | 2 +- packages/zowe-explorer/__mocks__/vscode.ts | 9 ++++++++- packages/zowe-explorer/src/dataset/DatasetTree.ts | 9 +++++++++ packages/zowe-explorer/src/job/ZosJobsProvider.ts | 9 +++++++++ packages/zowe-explorer/src/uss/USSTree.ts | 6 +++--- 5 files changed, 30 insertions(+), 5 deletions(-) diff --git a/packages/zowe-explorer/CHANGELOG.md b/packages/zowe-explorer/CHANGELOG.md index e73a25663c..60fc0a727c 100644 --- a/packages/zowe-explorer/CHANGELOG.md +++ b/packages/zowe-explorer/CHANGELOG.md @@ -14,7 +14,7 @@ All notable changes to the "vscode-extension-for-zowe" extension will be documen - Fixed issue where USS nodes were not removed from tree during deletion. [#2479](https://github.com/zowe/vscode-extension-for-zowe/issues/2479) - Fixed issue where new USS nodes from a paste operation were not shown in tree until refreshed. [#2479](https://github.com/zowe/vscode-extension-for-zowe/issues/2479) - Fixed issue where the "Delete Job" action showed a successful deletion message, even if the API returned an error. -- USS directories and sessions now update with their respective "collapsed icon" when collapsed. +- USS directories, PDS nodes, job nodes and session nodes now update with their respective "collapsed icon" when collapsed. ## `2.11.0` diff --git a/packages/zowe-explorer/__mocks__/vscode.ts b/packages/zowe-explorer/__mocks__/vscode.ts index c479dbf6c4..5cc8a04263 100644 --- a/packages/zowe-explorer/__mocks__/vscode.ts +++ b/packages/zowe-explorer/__mocks__/vscode.ts @@ -144,7 +144,12 @@ export namespace extensions { }; } } - +export interface TreeViewExpansionEvent { + /** + * Element that is expanded or collapsed. + */ + readonly element: T; +} export interface TreeView { /** * An optional human-readable message that will be rendered in the view. @@ -177,6 +182,8 @@ export interface TreeView { * **NOTE:** The {@link TreeDataProvider} that the `TreeView` {@link window.createTreeView is registered with} with must implement {@link TreeDataProvider.getParent getParent} method to access this API. */ reveal(element: T, options?: { select?: boolean; focus?: boolean; expand?: boolean | number }): Thenable; + + onDidCollapseElement: Event>; } export class FileDecoration { diff --git a/packages/zowe-explorer/src/dataset/DatasetTree.ts b/packages/zowe-explorer/src/dataset/DatasetTree.ts index 41e428768d..9a18aecca6 100644 --- a/packages/zowe-explorer/src/dataset/DatasetTree.ts +++ b/packages/zowe-explorer/src/dataset/DatasetTree.ts @@ -96,6 +96,15 @@ export class DatasetTree extends ZoweTreeProvider implements IZoweTree { + const newIcon = getIconByNode(e.element); + if (contextually.isPds(e.element) || contextually.isDsSession(e.element)) { + if (newIcon) { + e.element.iconPath = newIcon; + this.mOnDidChangeTreeData.fire(e.element); + } + } + }); } /** diff --git a/packages/zowe-explorer/src/job/ZosJobsProvider.ts b/packages/zowe-explorer/src/job/ZosJobsProvider.ts index 3333724e6f..06b4a65dcb 100644 --- a/packages/zowe-explorer/src/job/ZosJobsProvider.ts +++ b/packages/zowe-explorer/src/job/ZosJobsProvider.ts @@ -139,6 +139,15 @@ export class ZosJobsProvider extends ZoweTreeProvider implements IZoweTree { + const newIcon = getIconByNode(e.element); + if (contextually.isJob(e.element) || contextually.isJobsSession(e.element)) { + if (newIcon) { + e.element.iconPath = newIcon; + this.mOnDidChangeTreeData.fire(e.element); + } + } + }); } public rename(_node: IZoweJobTreeNode): void { diff --git a/packages/zowe-explorer/src/uss/USSTree.ts b/packages/zowe-explorer/src/uss/USSTree.ts index ed2d116179..bc7ba90d16 100644 --- a/packages/zowe-explorer/src/uss/USSTree.ts +++ b/packages/zowe-explorer/src/uss/USSTree.ts @@ -81,14 +81,14 @@ export class USSTree extends ZoweTreeProvider implements IZoweTree { - if (contextually.isUssDirectory(e.element)) { - const newIcon = getIconByNode(e.element); + const newIcon = getIconByNode(e.element); + if (contextually.isUssDirectory(e.element) || contextually.isUssSession(e.element)) { if (newIcon) { e.element.iconPath = newIcon; this.mOnDidChangeTreeData.fire(e.element); } } - }) + }); } /**